{
  "RepoName": "https://github.com/srittau/python-asserts.git",
  "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4",
  "Type": "logic error",
  "ErrorMessage": "======================================================================\nERROR: test_assert_greater_equal (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 579, in test_assert_greater_equal\n    assert_greater_equal(\"bar\", \"foo\")\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 529, in assert_greater_equal\n    if not int(first) >= int(second):\nValueError: invalid literal for int() with base 10: \"bar\"",
  "Issue": {
    "title": "Logical Error in assert_greater_equal Comparison",
    "description": "There is an issue in the `assert_greater_equal` function where the comparison between the first and second arguments is incorrectly cast to integers using `int()`. This can cause unexpected behavior when dealing with non-integer types, such as floats or strings, leading to incorrect assertion results. Users may observe assertion failures even when the first argument is indeed greater than or equal to the second argument, but type casting changes the outcome. This needs to be fixed to ensure proper functionality and accuracy of the `assert_greater_equal` function without unwanted type casting.",
    "explanation": "### Summary of the Issue and Solution\n\n#### Summary of the Issue:\nThe issue reported involves a logical error in the `assert_greater_equal` function. The function compares two arguments to determine if the first is greater than or equal to the second. However, the function incorrectly casts both arguments to integers before performing the comparison. This typecasting can lead to unexpected results when the arguments are not integers, such as when they are floats or strings. Consequently, valid comparisons might fail, resulting in incorrect assertion results. This issue affects the robustness and reliability of the `assert_greater_equal` function.\n\n#### Details of the Commit:\nThe commit provides a fix for the `assert_greater_equal` function by removing the unnecessary integer typecasting. The commit changes the function to perform a direct comparison of the original arguments without altering their types.\n\n#### Explanation of the Cause of the Issue:\nThe issue arises because the `assert_greater_equal` function forces both arguments into integer type before performing the comparison. This type coercion can alter the values of the arguments when they are not naturally integers, such as floats, strings, or other comparable types. For instance, the comparison between \"3\" (string) and 2 (integer) after converting both to integers yields a comparison between 3 and 2, which might still seem valid. However, comparing 2.9 (float) and 2 by casting them both to integers before the comparison would inaccurately reduce the float to 2, leading to an incorrect result.\n\n#### Solution Explained:\nThe commit addresses the issue by simplifying the comparison logic in `assert_greater_equal`. Instead of converting the arguments to integers, it retains their original types and compares them directly. This ensures that the function works correctly for any comparable types, such as floats or strings, without altering their values.\n\n### How the Commit Solves the Issue:\nBy removing the integer cast and comparing the arguments directly, the commit ensures that `assert_greater_equal` correctly reflects the comparison result of the original input types. This preserves the natural ordering and behavior of the types being compared.\n\n### Detailed Explanation of the Solution:\nTo better understand, let's break down the key improvements brought by this commit:\n\n1. **Type Integrity**: The commit ensures that the function respects the original data types of the arguments. This prevents unintended data loss or alteration caused by casting to integers.\n\n2. **Comprehensive Validation**: The updated function now correctly handles various types such as integers, floats, strings, or any other types that support comparison operations. This flexibility improves the utility and correctness of the function.\n\n3. **Improved Robustness**: By performing direct comparisons, the function avoids potential pitfalls associated with typecasting, such as truncating decimal values or changing string properties, which could previously lead to unexpected behavior or false assertion failures.\n\n### Conclusion:\nThe issue with `assert_greater_equal` was caused by inappropriate integer typecasting of arguments before comparison. This was resolved by the commit, which removes the unnecessary typecasting and allows direct comparison of the original values. This change makes the function more robust and reliable, capable of correctly handling a broader range of input types without introducing unexpected behavior."
  },
  "Explain": "### Summary of the Issue and Solution\n\n#### Summary of the Issue:\nThe issue reported involves a logical error in the `assert_greater_equal` function. The function compares two arguments to determine if the first is greater than or equal to the second. However, the function incorrectly casts both arguments to integers before performing the comparison. This typecasting can lead to unexpected results when the arguments are not integers, such as when they are floats or strings. Consequently, valid comparisons might fail, resulting in incorrect assertion results. This issue affects the robustness and reliability of the `assert_greater_equal` function.\n\n#### Details of the Commit:\nThe commit provides a fix for the `assert_greater_equal` function by removing the unnecessary integer typecasting. The commit changes the function to perform a direct comparison of the original arguments without altering their types.\n\n#### Explanation of the Cause of the Issue:\nThe issue arises because the `assert_greater_equal` function forces both arguments into integer type before performing the comparison. This type coercion can alter the values of the arguments when they are not naturally integers, such as floats, strings, or other comparable types. For instance, the comparison between \"3\" (string) and 2 (integer) after converting both to integers yields a comparison between 3 and 2, which might still seem valid. However, comparing 2.9 (float) and 2 by casting them both to integers before the comparison would inaccurately reduce the float to 2, leading to an incorrect result.\n\n#### Solution Explained:\nThe commit addresses the issue by simplifying the comparison logic in `assert_greater_equal`. Instead of converting the arguments to integers, it retains their original types and compares them directly. This ensures that the function works correctly for any comparable types, such as floats or strings, without altering their values.\n\n### How the Commit Solves the Issue:\nBy removing the integer cast and comparing the arguments directly, the commit ensures that `assert_greater_equal` correctly reflects the comparison result of the original input types. This preserves the natural ordering and behavior of the types being compared.\n\n### Detailed Explanation of the Solution:\nTo better understand, let's break down the key improvements brought by this commit:\n\n1. **Type Integrity**: The commit ensures that the function respects the original data types of the arguments. This prevents unintended data loss or alteration caused by casting to integers.\n\n2. **Comprehensive Validation**: The updated function now correctly handles various types such as integers, floats, strings, or any other types that support comparison operations. This flexibility improves the utility and correctness of the function.\n\n3. **Improved Robustness**: By performing direct comparisons, the function avoids potential pitfalls associated with typecasting, such as truncating decimal values or changing string properties, which could previously lead to unexpected behavior or false assertion failures.\n\n### Conclusion:\nThe issue with `assert_greater_equal` was caused by inappropriate integer typecasting of arguments before comparison. This was resolved by the commit, which removes the unnecessary typecasting and allows direct comparison of the original values. This change makes the function more robust and reliable, capable of correctly handling a broader range of input types without introducing unexpected behavior.",
  "Time": "2024-06-10",
  "Difficulty": "Medium",
  "OriginCode": [
    {
      "path": "srittau_python-asserts/pyproject.toml",
      "content": "[tool.poetry]\nname = \"asserts\"\nversion = \"0.13.2.dev0\"\ndescription = \"Stand-alone Assertions\"\nreadme = \"README.md\"\nauthors = [\"Sebastian Rittau <srittau@rittau.biz>\"]\nlicense = \"MIT\"\nhomepage = \"https://github.com/srittau/python-asserts\"\nrepository = \"https://github.com/srittau/python-asserts\"\nclassifiers = [\n    \"Development Status :: 3 - Alpha\",\n    \"Intended Audience :: Developers\",\n    \"Topic :: Software Development :: Quality Assurance\",\n    \"Topic :: Software Development :: Testing\",\n]\npackages = [{ include = \"asserts\" }]\ninclude = [\"*/py.typed\", \"*.pyi\"]\n\n[tool.poetry.urls]\n\"GitHub\" = \"https://github.com/srittau/python-asserts\"\n\"Bug Tracker\" = \"https://github.com/srittau/python-asserts/issues\"\n\"Changes\" = \"https://github.com/srittau/python-asserts/blob/main/CHANGELOG.md\"\n\n[tool.poetry.dependencies]\npython = \">=3.8.1\"\ntyping-extensions = \"^4.10.0\"\n\n[tool.poetry.group.dev.dependencies]\nmypy = \">=1.9,<1.11\"\npoethepoet = \">=0.25,<0.27\"\nruff = \">=0.3.2,<0.5.0\"\n\n[tool.ruff]\nline-length = 79\ntarget-version = \"py38\"\n\n[tool.poe.tasks]\ntest = \"python3 -Wall -m unittest test_asserts\"\ndoctest = \"python3 -m doctest asserts/__init__.py\"\nlint = \"ruff check asserts test_asserts.py\"\ntypecheck = \"mypy asserts test_asserts.py\"\n\n[build-system]\nrequires = [\"poetry-core>=1.0.0\"]\nbuild-backend = \"poetry.core.masonry.api\"\n"
    },
    {
      "path": "srittau_python-asserts/test_asserts.py",
      "content": "# -*- coding: utf-8 -*-\n\nimport re\nimport sys\nfrom collections import OrderedDict\nfrom datetime import datetime, timedelta, timezone\nfrom json import JSONDecodeError\nfrom unittest import TestCase\nfrom warnings import catch_warnings, simplefilter, warn\n\nfrom asserts import (\n    Absent,\n    Exists,\n    Present,\n    assert_almost_equal,\n    assert_between,\n    assert_boolean_false,\n    assert_boolean_true,\n    assert_count_equal,\n    assert_datetime_about_now,\n    assert_datetime_about_now_utc,\n    assert_dict_equal,\n    assert_dict_superset,\n    assert_equal,\n    assert_false,\n    assert_greater,\n    assert_greater_equal,\n    assert_has_attr,\n    assert_in,\n    assert_is,\n    assert_is_instance,\n    assert_is_none,\n    assert_is_not,\n    assert_is_not_none,\n    assert_json_subset,\n    assert_less,\n    assert_less_equal,\n    assert_not_almost_equal,\n    assert_not_equal,\n    assert_not_in,\n    assert_not_is_instance,\n    assert_not_regex,\n    assert_raises,\n    assert_raises_errno,\n    assert_raises_regex,\n    assert_regex,\n    assert_succeeds,\n    assert_true,\n    assert_warns,\n    assert_warns_regex,\n    fail,\n)\n\n\nclass Box:\n    def __init__(self, initial_value):\n        self.value = initial_value\n\n\nclass _DummyObject(object):\n    def __init__(self, value=\"x\"):\n        self.value = value\n\n    def __repr__(self):\n        return \"<Dummy>\"\n\n\ndef _assert_raises_assertion(expected_message):\n    \"\"\"Fail if the context does not raise an AssertionError or the exception\n    message does not match.\n\n    This is used to test assertions, without using those assertions.\n\n    \"\"\"\n\n    class Context(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type is None:\n                raise AssertionError(\"no AssertionError raised\")\n            if exc_type != AssertionError:\n                return False\n            if str(exc_val) != expected_message:\n                raise AssertionError(\n                    \"expected exception message {!r}, got {!r}\".format(\n                        expected_message, str(exc_val)\n                    )\n                )\n            return True\n\n    return Context()\n\n\nclass AssertTest(TestCase):\n    _type_string = \"type\" if sys.version_info[0] < 3 else \"class\"\n\n    # fail()\n\n    def test_fail__default_message(self):\n        with _assert_raises_assertion(\"assertion failure\"):\n            fail()\n\n    def test_fail__with_message(self):\n        with _assert_raises_assertion(\"test message\"):\n            fail(\"test message\")\n\n    # assert_true()\n\n    def test_assert_true__truthy_value(self):\n        assert_true(\"Hello World!\")\n\n    def test_assert_true__falsy_value__default_message(self):\n        with _assert_raises_assertion(\"'' is not truthy\"):\n            assert_true(\"\")\n\n    def test_assert_true__falsy_value__custom_message(self):\n        with _assert_raises_assertion(\"0 is not truthy;0\"):\n            assert_true(0, \"{msg};{expr}\")\n\n    # assert_false()\n\n    def test_assert_false__falsy_value(self):\n        assert_false(\"\")\n\n    def test_assert_false__truthy_value__default_message(self):\n        with _assert_raises_assertion(\"25 is not falsy\"):\n            assert_false(25)\n\n    def test_assert_false__truthy_value__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is not falsy;foo\"):\n            assert_false(\"foo\", \"{msg};{expr}\")\n\n    # assert_boolean_true()\n\n    def test_assert_boolean_true__true(self):\n        assert_boolean_true(True)\n\n    def test_assert_boolean_true__false__custom_message(self):\n        with _assert_raises_assertion(\"'Foo' is not True;Foo\"):\n            assert_boolean_true(\"Foo\", \"{msg};{expr}\")\n\n    def test_assert_boolean_true__truthy__default_message(self):\n        with _assert_raises_assertion(\"1 is not True\"):\n            assert_boolean_true(1)\n\n    # assert_boolean_false()\n\n    def test_assert_boolean_false__false(self):\n        assert_boolean_false(False)\n\n    def test_assert_boolean_false__true__default_message(self):\n        with _assert_raises_assertion(\"'foo' is not False\"):\n            assert_boolean_false(\"foo\")\n\n    def test_assert_boolean_false__falsy__custom_message(self):\n        with _assert_raises_assertion(\"0 is not False;0\"):\n            assert_boolean_false(0, \"{msg};{expr}\")\n\n    # assert_is_none()\n\n    def test_assert_is_none__none(self):\n        assert_is_none(None)\n\n    def test_assert_is_none__string__default_message(self):\n        with _assert_raises_assertion(\"'' is not None\"):\n            assert_is_none(\"\")\n\n    def test_assert_is_none__int__custom_message(self):\n        with _assert_raises_assertion(\"55 is not None;55\"):\n            assert_is_none(55, \"{msg};{expr}\")\n\n    # assert_is_not_none()\n\n    def test_assert_is_not_none__string(self):\n        assert_is_not_none(\"\")\n\n    def test_assert_is_not_none__none__default_message(self):\n        with _assert_raises_assertion(\"expression is None\"):\n            assert_is_not_none(None)\n\n    def test_assert_is_not_none__none__custom_message(self):\n        with _assert_raises_assertion(\"expression is None;None\"):\n            assert_is_not_none(None, \"{msg};{expr!r}\")\n\n    # assert_equal()\n\n    def test_assert_equal__equal_strings(self):\n        assert_equal(\"foo\", \"foo\")\n\n    def test_assert_equal__equal_objects(self):\n        class MyClass(object):\n            def __eq__(self, other):\n                return True\n\n        assert_equal(MyClass(), MyClass())\n\n    def test_assert_equal__not_equal__default_message(self):\n        with _assert_raises_assertion(\"'string' != 55\"):\n            assert_equal(\"string\", 55)\n\n    def test_assert_equal__not_equal__custom_message(self):\n        with _assert_raises_assertion(\"'string' != 55;'string';55\"):\n            assert_equal(\"string\", 55, \"{msg};{first!r};{second!r}\")\n\n    def test_assert_equal__dict(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_equal({\"foo\": 5}, {})\n\n    # assert_not_equal()\n\n    def test_assert_not_equal__not_equal(self):\n        assert_not_equal(\"abc\", \"def\")\n\n    def test_assert_not_equal__equal__default_message(self):\n        with _assert_raises_assertion(\"'abc' == 'abc'\"):\n            assert_not_equal(\"abc\", \"abc\")\n\n    def test_assert_not_equal__equal__custom_message(self):\n        with _assert_raises_assertion(\"1.0 == 1;1.0;1\"):\n            assert_not_equal(1.0, 1, \"{msg};{first};{second}\")\n\n    # assert_almost_equal()\n\n    def test_assert_almost_equal__same(self):\n        assert_almost_equal(5, 5)\n\n    def test_assert_almost_equal__similar__defaults(self):\n        assert_almost_equal(5, 5.00000001)\n\n    def test_assert_almost_equal__similar__places(self):\n        assert_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_almost_equal__similar__delta(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__similar__delta_reverse(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__default_message(self):\n        with _assert_raises_assertion(\"5 != 5.0001 within 7 places\"):\n            assert_almost_equal(5, 5.0001)\n\n    def test_assert_almost_equal__not_similar__places__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 within 3 places\"):\n            assert_almost_equal(5, 6, places=3)\n\n    def test_assert_almost_equal__not_similar__delta__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1\"):\n            assert_almost_equal(5, 6, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__delta_reverse(self):\n        with _assert_raises_assertion(\"6 != 5 with delta=0.3\"):\n            assert_almost_equal(6, 5, delta=0.3)\n\n    def test_assert_almost_equal__not_similar__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 7 places;5;-5;7;None\"):\n            assert_almost_equal(\n                5, -5, msg_fmt=\"{msg};{first};{second};{places};{delta!r}\"\n            )\n\n    def test_assert_almost_equal__not_similar__places__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 3 places;5;-5;3;None\"):\n            assert_almost_equal(\n                5,\n                -5,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_almost_equal__not_similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1;5;6;None;0.1\"):\n            assert_almost_equal(\n                5,\n                6,\n                delta=0.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_almost_equal__wrong_types(self):\n        try:\n            assert_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__places_and_delta(self):\n        try:\n            assert_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_not_almost_equal()\n\n    def test_assert_not_almost_equal__same(self):\n        with _assert_raises_assertion(\"5 == 5 within 7 places\"):\n            assert_not_almost_equal(5, 5)\n\n    def test_assert_not_almost_equal__similar__defaults(self):\n        with _assert_raises_assertion(\"5 == 5.00000001 within 7 places\"):\n            assert_not_almost_equal(5, 5.00000001)\n\n    def test_assert_not_almost_equal__similar__places(self):\n        with _assert_raises_assertion(\"5 == 5.0001 within 3 places\"):\n            assert_not_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_not_almost_equal__similar__delta(self):\n        with _assert_raises_assertion(\"5 == 5.1 with delta=0.1\"):\n            assert_not_almost_equal(5, 5.1, delta=0.1)\n\n    def test_assert_not_almost_equal__similar__delta_reverse(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.3\"):\n            assert_almost_equal(5, 6, delta=0.3)\n\n    def test_assert_not_almost_equal__not_similar(self):\n        assert_not_almost_equal(5, 5.0001)\n\n    def test_assert_not_almost_equal__not_similar__delta(self):\n        assert_not_almost_equal(5, 5.1, delta=0.05)\n\n    def test_assert_not_almost_equal__not_similar__delta_reverse(self):\n        assert_not_almost_equal(5.1, 5, delta=0.05)\n\n    def test_assert_not_almost_equal__similar__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.00000001 within 7 places;5;5.00000001;7;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.00000001,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__places__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.0001 within 3 places;5;5.0001;3;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.0001,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 == 6 with delta=1.1;5;6;None;1.1\"):\n            assert_not_almost_equal(\n                5,\n                6,\n                delta=1.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_not_almost_equal__wrong_types(self):\n        try:\n            assert_not_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_not_almost_equal__places_and_delta(self):\n        try:\n            assert_not_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_not_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_not_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_dict_equal()\n\n    def test_assert_dict_equal__empty_dicts(self):\n        assert_dict_equal({}, {})\n\n    def test_assert_dict_equal__dicts_are_equal(self):\n        assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_equal__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_equal({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_equal__one_key_missing_from_left(self):\n        with _assert_raises_assertion(\"extra key 'foo' in right dict\"):\n            assert_dict_equal({\"bar\": 10}, {\"bar\": 10, \"foo\": 5})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_left(self):\n        with _assert_raises_assertion(\"extra keys 'bar', 'foo' in right dict\"):\n            assert_dict_equal({\"baz\": 15}, {\"foo\": 5, \"bar\": 10, \"baz\": 15})\n\n    def test_assert_dict_equal__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_equal({\"foo\": 15}, {\"foo\": 10})\n\n    def test_assert_dict_equal__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_equal({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_equal({5: \"\", \"foo\": \"\"}, {})\n        with _assert_raises_assertion(\"extra key 10 in right dict\"):\n            assert_dict_equal({}, {10: \"foo\"})\n        with _assert_raises_assertion(\"extra keys 'foo', 5 in right dict\"):\n            assert_dict_equal({}, {5: \"\", \"foo\": \"\"})\n\n    def test_assert_dict_equal__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal(\n                {\"foo\": \"\", \"bar\": \"\", \"baz\": 5},\n                {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"},\n            )\n        with _assert_raises_assertion(\"extra key 'extra' in right dict\"):\n            assert_dict_equal(\n                {\"bar\": \"\", \"baz\": 5}, {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"}\n            )\n\n    def test_assert_dict_equal__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo'];['bar']\"\n        ):\n            assert_dict_equal(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{missing_keys!r};{extra_keys!r}\",\n            )\n\n    def test_assert_dict_equal__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_equal(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_dict_superset()\n\n    def test_assert_dict_superset__empty_dicts(self):\n        assert_dict_superset({}, {})\n\n    def test_assert_dict_superset__dicts_are_equal(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_superset__dicts_is_superset(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n\n    def test_assert_dict_superset__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_superset__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_superset({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_superset__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_superset({\"foo\": 15}, {\"foo\": 10, \"bar\": 15})\n\n    def test_assert_dict_superset__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_superset({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_superset({5: \"\", \"foo\": \"\"}, {})\n\n    def test_assert_dict_superset__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"foo\": \"\", \"bar\": 5}, {\"bar\": 1})\n\n    def test_assert_dict_superset__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo']\"\n        ):\n            assert_dict_superset(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\" \"{missing_keys!r}\",\n            )\n\n    def test_assert_dict_superset__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_superset(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_less()\n\n    def test_assert_less(self):\n        assert_less(4, 5)\n        with _assert_raises_assertion(\"5 is not less than 5\"):\n            assert_less(5, 5)\n        with _assert_raises_assertion(\"'foo' is not less than 'bar'\"):\n            assert_less(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than 5;6;5\"):\n            assert_less(6, 5, \"{msg};{first};{second}\")\n\n    # assert_less_equal()\n\n    def test_assert_less_equal(self):\n        assert_less_equal(4, 5)\n        assert_less_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'foo' is not less than or equal to 'bar'\"\n        ):\n            assert_less_equal(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than or equal to 5;6;5\"):\n            assert_less_equal(6, 5, \"{msg};{first};{second}\")\n\n    # assert_greater()\n\n    def test_assert_greater(self):\n        assert_greater(5, 4)\n        with _assert_raises_assertion(\"5 is not greater than 5\"):\n            assert_greater(5, 5)\n        with _assert_raises_assertion(\"'bar' is not greater than 'foo'\"):\n            assert_greater(\"bar\", \"foo\")\n        with _assert_raises_assertion(\"5 is not greater than 6;5;6\"):\n            assert_greater(5, 6, \"{msg};{first};{second}\")\n\n    # assert_greater_equal()\n\n    def test_assert_greater_equal(self):\n        assert_greater_equal(5, 4)\n        assert_greater_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'bar' is not greater than or equal to 'foo'\"\n        ):\n            assert_greater_equal(\"bar\", \"foo\")\n        with _assert_raises_assertion(\n            \"5 is not greater than or equal to 6;5;6\"\n        ):\n            assert_greater_equal(5, 6, \"{msg};{first};{second}\")\n\n    # assert_regex()\n\n    def test_assert_regex__matches_string(self):\n        assert_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_regex__matches_regex(self):\n        regex = re.compile(\"is.*test\")\n        assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", \"not found\")\n\n    def test_assert_regex__does_not_match_regex__default_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';\"\n            \"'Wrong text';'not found'\"\n        ):\n            assert_regex(\n                \"Wrong text\", r\"not found\", \"{msg};{text!r};{pattern!r}\"\n            )\n\n    def test_assert_regex__does_not_match_regex__custom_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';'Wrong text';\"\n            \"'not found'\"\n        ):\n            assert_regex(\"Wrong text\", regex, \"{msg};{text!r};{pattern!r}\")\n\n    # assert_not_regex()\n\n    def test_assert_not_regex__does_not_match_string(self):\n        assert_not_regex(\"This is a test text\", \"no match\")\n\n    def test_assert_not_regex__does_not_match_regex(self):\n        regex = re.compile(\"no match\")\n        assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_not_regex__matches_regex__default_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';\"\n            \"'This is a test text';'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\",\n                \"is.*test\",\n                \"{msg};{text!r};{pattern!r}\",\n            )\n\n    def test_assert_not_regex__matches_regex__custom_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';'This is a test text';\"\n            \"'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\", regex, \"{msg};{text!r};{pattern!r}\"\n            )\n\n    # assert_is()\n\n    def test_assert_is__same(self):\n        x = _DummyObject()\n        assert_is(x, x)\n\n    def test_assert_is__not_same__default_message(self):\n        with _assert_raises_assertion(\"'x' is not 'y'\"):\n            assert_is(\"x\", \"y\")\n\n    def test_assert_is__equal_but_not_same__custom_message(self):\n        x = \"x\"\n        y = _DummyObject(\"y\")\n        with _assert_raises_assertion(\"'x' is not <Dummy>;'x';y\"):\n            assert_is(x, y, \"{msg};{first!r};{second.value}\")\n\n    # assert_is_not()\n\n    def test_assert_is_not__not_same(self):\n        x = _DummyObject()\n        y = _DummyObject()\n        assert_is_not(x, y)\n\n    def test_assert_is_not__same__default_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>\"):\n            assert_is_not(x, x)\n\n    def test_assert_is_not__same__custom_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>;x;x\"):\n            assert_is_not(x, x, \"{msg};{first.value};{second.value}\")\n\n    # assert_in()\n\n    def test_assert_in__contains(self):\n        assert_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_in__does_not_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' not in []\"):\n            assert_in(\"foo\", [])\n\n    def test_assert_in__does_not_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' not in [];'foo';[]\"):\n            assert_in(\"foo\", [], \"{msg};{first!r};{second!r}\")\n\n    # assert_not_in()\n\n    def test_assert_not_in__does_not_contain(self):\n        assert_not_in(\"foo\", [])\n\n    def test_assert_not_in__does_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar', 'baz']\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_not_in__does_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar'];'foo';bar\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\"], \"{msg};{first!r};{second[1]}\")\n\n    # assert_count_equal()\n\n    def test_assert_count_equal__equal(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], [\"a\"])\n\n    def test_assert_count_equal__equal_differing_types(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], {\"a\"})\n\n    def test_assert_count_equal__ignore_order(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\", \"b\"], [\"b\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal([], {\"a\"})\n\n    def test_assert_count_equal__multiple_missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'b', 'c'\"):\n            assert_count_equal([\"a\"], [\"a\", \"b\", \"c\"])\n\n    def test_assert_count_equal__respect_duplicates(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal({\"a\"}, [\"a\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence2(self):\n        with _assert_raises_assertion(\"missing from sequence 2: 'a', 'c'\"):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"b\"])\n\n    def test_assert_count_equal__missing_from_both(self):\n        msg = \"missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\"\n        with _assert_raises_assertion(msg):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n\n    def test_assert_count_equal__custom_message(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a';[];['a']\"):\n            assert_count_equal([], [\"a\"], \"{msg};{first};{second}\")\n\n    # assert_between()\n\n    def test_assert_between__within_range(self):\n        assert_between(0, 10, 0)\n        assert_between(0, 10, 10)\n        assert_between(0, 10, 5)\n\n    def test_assert_between__too_low__default_message(self):\n        with _assert_raises_assertion(\"-1 is not between 0 and 10\"):\n            assert_between(0, 10, -1)\n\n    def test_assert_between__too_high__custom_message(self):\n        with _assert_raises_assertion(\"11 is not between 0 and 10;0;10;11\"):\n            assert_between(0, 10, 11, \"{msg};{lower};{upper};{expr}\")\n\n    # assert_is_instance()\n\n    def _is_instance_message(self, expr, expected_type, real_type):\n        expected_message = (\n            \"{!r} is an instance of <class {}>, expected {}\".format(\n                expr, real_type, expected_type\n            )\n        )\n        if sys.version_info[0] < 3:\n            return expected_message.replace(\"class\", \"type\")\n        else:\n            return expected_message\n\n    def test_assert_is_instance__single_type(self):\n        assert_is_instance(4, int)\n        assert_is_instance(OSError(), Exception)\n\n    def test_assert_is_instance__multiple_types(self):\n        assert_is_instance(4, (str, int))\n\n    def test_assert_is_instance__default_message(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        with _assert_raises_assertion(expected_message):\n            assert_is_instance(\"my string\", int)\n\n    def test_assert_is_instance__custom_message_single_type(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>,)\".format(expected_message)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\"my string\", int, \"{msg};{obj};{types}\")\n\n    def test_assert_is_instance__custom_message_multiple_types(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"(<class 'int'>, <class 'float'>)\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>, <class 'float'>)\".format(\n            expected_message\n        )\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\n                \"my string\", (int, float), \"{msg};{obj};{types}\"\n            )\n\n    # assert_not_is_instance()\n\n    def _not_is_instance_message(self, obj):\n        expected_message = \"{!r} is an instance of {}\".format(\n            obj, obj.__class__\n        )\n        if sys.version_info[0] < 3:\n            expected_message = expected_message.replace(\"class\", \"type\")\n            expected_message = expected_message.replace(\n                \"type 'OSError'\", \"type 'exceptions.OSError'\"\n            )\n        return expected_message\n\n    def test_assert_not_is_instance__single_type(self):\n        assert_not_is_instance(4, str)\n\n    def test_assert_not_is_instance__multiple_types(self):\n        assert_not_is_instance(4, (str, bytes))\n\n    def test_assert_not_is_instance__default_message(self):\n        obj = OSError()\n        expected_message = self._not_is_instance_message(obj)\n        with _assert_raises_assertion(expected_message):\n            assert_not_is_instance(obj, Exception)\n\n    def test_assert_not_is_instance__custom_message__single_type(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>,)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", str, \"{msg};{obj};{types!r}\")\n\n    def test_assert_not_is_instance__custom_message__multiple_types(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>, <class 'int'>)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", (str, int), \"{msg};{obj};{types!r}\")\n\n    # assert_has_attr()\n\n    def test_assert_has_attr__has_attribute(self):\n        d = _DummyObject()\n        assert_has_attr(d, \"value\")\n\n    def test_assert_has_attr__does_not_have_attribute__default_message(self):\n        d = _DummyObject()\n        with _assert_raises_assertion(\"<Dummy> does not have attribute 'foo'\"):\n            assert_has_attr(d, \"foo\")\n\n    def test_assert_has_attr__does_not_have_attribute__custom_message(self):\n        d = _DummyObject()\n        expected = \"<Dummy> does not have attribute 'foo';<Dummy>;foo\"\n        with _assert_raises_assertion(expected):\n            assert_has_attr(d, \"foo\", msg_fmt=\"{msg};{obj!r};{attribute}\")\n\n    # assert_datetime_about_now()\n\n    def test_assert_datetime_about_now__close(self):\n        assert_datetime_about_now(datetime.now())\n\n    def test_assert_datetime_about_now__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(None)\n\n    def test_assert_datetime_about_now__none__custom_message(self):\n        dt = datetime.now().date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now__too_low(self):\n        then = datetime.now() - timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__too_high(self):\n        then = datetime.now() + timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"^datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            \"is not close to current date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now().date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_datetime_about_now_utc()\n\n    def test_assert_datetime_about_now_utc__close(self):\n        assert_datetime_about_now_utc(\n            datetime.now(timezone.utc).replace(tzinfo=None)\n        )\n\n    def test_assert_datetime_about_now_utc__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(None)\n\n    def test_assert_datetime_about_now_utc__none__custom_message(self):\n        dt = datetime.now(timezone.utc).date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now_utc__too_low(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) - timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__too_high(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) + timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            r\"is not close to current UTC date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now(timezone.utc).date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current UTC date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_raises()\n\n    def test_assert_raises__raises_right_exception(self):\n        with assert_raises(KeyError):\n            raise KeyError()\n\n    def test_assert_raises__exc_val(self):\n        exc = KeyError()\n        with assert_raises(KeyError) as context:\n            raise exc\n        assert_is(exc, context.exc_val)\n\n    def test_assert_raises__exc_val_within_context(self):\n        with assert_raises(RuntimeError):\n            with assert_raises(KeyError) as context:\n                context.exc_val\n\n    def test_assert_raises__raises_subclass(self):\n        class MyError(IndexError):\n            pass\n\n        with assert_raises(IndexError):\n            raise MyError()\n\n    def test_assert_raises__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises(KeyError):\n                pass\n\n    def test_assert_raises__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError\"\n        with _assert_raises_assertion(expected):\n            with assert_raises(\n                KeyError, msg_fmt=\"{msg};{exc_type.__name__};{exc_name}\"\n            ):\n                pass\n\n    def test_assert_raises__wrong_exception_raised(self):\n        try:\n            with assert_raises(IndexError):\n                raise KeyError()\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(exc):\n            assert_is_instance(exc, KeyError)\n            called.value = True\n\n        with assert_raises(KeyError) as context:\n            context.add_test(extra_test)\n            raise KeyError()\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_raises__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_):\n            called.value = True\n\n        with assert_raises(AssertionError):\n            with assert_raises(KeyError) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_raises_regex()\n\n    def test_assert_raises_regex__raises_right_exception(self):\n        with assert_raises_regex(KeyError, r\"test.*\"):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__raises_right_exception__compiled(self):\n        with assert_raises_regex(KeyError, re.compile(r\"test.*\")):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                pass\n\n    def test_assert_raises_regex__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError;'';test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_regex__no_message__default_message(self):\n        with _assert_raises_assertion(\"KeyError without message\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError()\n\n    def test_assert_raises_regex__no_message__custom_message(self):\n        expected = \"KeyError without message;KeyError;KeyError;None;test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError()\n\n    def test_assert_raises_regex__wrong_exception_raised(self):\n        try:\n            with assert_raises_regex(IndexError, \"test message\"):\n                raise KeyError(\"test message\")\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises_regex__wrong_error__default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__pattern_default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, re.compile(r\"test\")):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__custom_message(self):\n        expected = (\n            \"'wrong message' does not match 'test';KeyError;KeyError;\"\n            \"'wrong message';test\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError(\"wrong message\")\n\n    # assert_raises_errno()\n\n    def test_assert_raises_errno__right_errno(self):\n        with assert_raises_errno(OSError, 20):\n            raise OSError(20, \"Test error\")\n\n    def test_assert_raises_errno__no_exception_raised__default_message(self):\n        with _assert_raises_assertion(\"OSError not raised\"):\n            with assert_raises_errno(OSError, 20):\n                pass\n\n    def test_assert_raises_errno__no_exception_raised__custom_message(self):\n        expected = \"OSError not raised;OSError;OSError;20;None\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_errno__wrong_class_raised(self):\n        class RightClass(OSError):\n            pass\n\n        class WrongClass(OSError):\n            pass\n\n        try:\n            with assert_raises_errno(RightClass, 20):\n                raise WrongClass(20, \"Test error\")\n        except WrongClass:\n            pass\n        else:\n            raise AssertionError(\"WrongClass was not raised\")\n\n    def test_assert_raises_errno__wrong_errno__default_message(self):\n        with _assert_raises_assertion(\"wrong errno: 20 != 1\"):\n            with assert_raises_errno(OSError, 20):\n                raise OSError(1, \"Test error\")\n\n    def test_assert_raises_errno__wrong_errno__custom_message(self):\n        expected = \"wrong errno: 20 != 1;OSError;OSError;20;1\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                raise OSError(1, \"Test error\")\n\n    # assert_succeeds()\n\n    def test_assert_succeeds__no_exception_raised(self):\n        with assert_succeeds(KeyError):\n            pass\n\n    def test_assert_succeeds__expected_exception__default_message(self):\n        with _assert_raises_assertion(\"KeyError was unexpectedly raised\"):\n            with assert_succeeds(KeyError):\n                raise KeyError()\n\n    def test_assert_succeeds__expected_exception__custom_message(self):\n        expected = (\n            \"KeyError was unexpectedly raised;KeyError;KeyError;test error\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{exception.args[0]}\"\n            )\n            with assert_succeeds(KeyError, msg_fmt=msg_fmt):\n                raise KeyError(\"test error\")\n\n    def test_assert_succeeds__unexpected_exception(self):\n        try:\n            with assert_succeeds(ValueError):\n                raise KeyError()\n        except KeyError:\n            pass\n        else:\n            raise AssertionError(\"KeyError was not raised\")\n\n    # assert_warns()\n\n    def test_assert_warns__default_message(self):\n        with assert_raises_regex(AssertionError, r\"^ImportWarning not issued\"):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__custom_message(self):\n        exception = \"ImportWarning not issued;ImportWarning;ImportWarning\"\n        with _assert_raises_assertion(exception):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name}\"\n            with assert_warns(ImportWarning, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(FutureWarning):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(UserWarning):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns(UserWarning):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns(UserWarning):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(warning):\n            assert_is(warning.category, UserWarning)\n            called.value = True\n            return True\n\n        with assert_warns(UserWarning) as context:\n            context.add_test(extra_test)\n            warn(\"bar\", UserWarning)\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_warns__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_: Warning) -> bool:\n            called.value = True\n            return False\n\n        with assert_raises(AssertionError):\n            with assert_warns(UserWarning) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_warns_regex()\n\n    def test_assert_warns_regex__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"fo+\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__warning_text_matches_in_the_middle(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"o\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                pass\n\n    def test_assert_warns_regex__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(ImportWarning, r\"foo\"):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns_regex__wrong_message(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UnicodeWarning, r\"foo\"):\n                warn(\"bar\", UnicodeWarning)\n\n    def test_assert_warns_regex__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(UserWarning, r\"bar2\"):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar1\", UserWarning)\n                warn(\"bar2\", UserWarning)\n                warn(\"bar3\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns_regex(UserWarning, r\"\"):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__not_issued__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__not_issued__custom_message(self):\n        expected = \"no ImportWarning matching 'abc' issued;ImportWarning;ImportWarning;abc\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(ImportWarning, r\"abc\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns_regex__wrong_message__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__wrong_message__custom_message(self):\n        expected = (\n            \"no UserWarning matching 'foo.*bar' issued;UserWarning;\"\n            \"UserWarning;foo.*bar\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(UserWarning, r\"foo.*bar\", msg_fmt=msg_fmt):\n                pass\n\n    # assert_json_subset()\n\n    def test_assert_json_subset__different_types(self):\n        with _assert_raises_assertion(\"element $ differs: {} != []\"):\n            assert_json_subset({}, [])\n\n    def test_assert_json_subset__empty_objects(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, {})\n\n    def test_assert_json_subset__objects_equal(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(\n                {\"foo\": 3, \"bar\": \"abc\"}, {\"bar\": \"abc\", \"foo\": 3}\n            )\n\n    def test_assert_json_subset__one_key_missing_from_first_object(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n\n    def test_assert_json_subset__one_key_missing_from_second_object(self):\n        with _assert_raises_assertion(\"element 'bar' missing from element $\"):\n            assert_json_subset({\"foo\": 3, \"bar\": 3}, {\"foo\": 3})\n\n    def test_assert_json_subset__multiple_keys_missing_from_second_object(\n        self,\n    ):\n        with _assert_raises_assertion(\n            \"elements 'bar', 'baz', and 'foo' missing from element $\"\n        ):\n            assert_json_subset({\"foo\": 3, \"bar\": 3, \"baz\": 3}, {})\n\n    def test_assert_json_subset__value_differs(self):\n        with _assert_raises_assertion(\"element $['foo'] differs: 3 != 4\"):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n\n    def test_assert_json_subset__empty_lists(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([], [])\n\n    def test_assert_json_subset__different_sized_lists(self):\n        with _assert_raises_assertion(\"JSON array $ differs in size: 2 != 1\"):\n            assert_json_subset([1, 2], [1])\n        with _assert_raises_assertion(\"JSON array $ differs in size: 1 != 2\"):\n            assert_json_subset([1], [1, 2])\n\n    def test_assert_json_subset__different_list_values(self):\n        with _assert_raises_assertion(\"element $[0] differs: {} != []\"):\n            assert_json_subset([{}], [[]])\n\n    def test_assert_json_subset__fundamental_types_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 1 != 'foo'\"):\n            assert_json_subset([1], [\"foo\"])\n\n    def test_assert_json_subset__fundamental_values_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 'bar' != 'foo'\"):\n            assert_json_subset([\"bar\"], [\"foo\"])\n\n    def test_assert_json_subset__none(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([None], [None])\n        with _assert_raises_assertion(\"element $[0] differs: 42 != None\"):\n            assert_json_subset([42], [None])\n        with _assert_raises_assertion(\"element $[0] differs: None != 42\"):\n            assert_json_subset([None], [42])\n\n    def test_assert_json_subset__compare_int_and_float(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([42], [42.0])\n            assert_json_subset([42.0], [42])\n\n    def test_assert_json_subset__unsupported_type(self):\n        msg = \"unsupported type <{} 'set'>\".format(self._type_string)\n        with assert_raises_regex(TypeError, msg):\n            assert_json_subset([set()], [set()])\n\n    def test_assert_json_subset__subtypes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(OrderedDict(), {})\n            assert_json_subset({}, OrderedDict())\n\n    def test_assert_json_subset__second_is_string(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, \"{  }\")\n\n    def test_assert_json_subset__second_is_unsupported_json_string(self):\n        msg = \"second must decode to dict or list, not <{} 'int'>\".format(\n            self._type_string\n        )\n        with _assert_raises_assertion(msg):\n            assert_json_subset({}, \"42\")\n\n    def test_assert_json_subset__second_is_invalid_json_string(self):\n        with assert_raises(JSONDecodeError):\n            assert_json_subset({}, \",\")\n\n    def test_assert_json_subset__second_is_bytes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"utf-8\"))\n\n    def test_assert_json_subset__second_is_latin1_bytes(self):\n        with assert_raises(UnicodeDecodeError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"iso-8859-1\"))\n\n    def test_assert_json_subset__invalid_type(self):\n        with assert_raises_regex(\n            TypeError, \"second must be dict, list, str, or bytes\"\n        ):\n            assert_json_subset({}, 42)  # type: ignore[arg-type]\n\n    def test_assert_json_subset__element_name_not_str(self) -> None:\n        with assert_raises_regex(\n            TypeError,\n            \"12 is not a valid object member name\",\n        ):\n            assert_json_subset({12: 34}, \"{}\")\n\n    def test_assert_json_subset__presence_check(self) -> None:\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present()}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present()}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent()}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent()}, {\"foo\": \"bar\"})\n\n    def test_assert_json_subset__existence_check_old(self) -> None:\n        with catch_warnings():\n            simplefilter(\"ignore\")\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): True}, {\"foo\": \"bar\"})\n            with assert_raises_regex(\n                AssertionError,\n                r\"element 'foo' missing from element \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): True}, {})\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): False}, {})\n            with assert_raises_regex(\n                AssertionError,\n                r\"spurious member 'foo' in object \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): False}, {\"foo\": \"bar\"})\n"
    },
    {
      "path": "srittau_python-asserts/CHANGELOG.md",
      "content": "# Changelog for python-asserts\n\npython-asserts adheres to [semantic versioning](https://semver.org/).\n\n## UNRELEASED –\n\n## [0.13.1] – 2024-04-29\n\n### Fixed\n\nFixed Python 3.12 deprecation warnings.\n\n## [0.13.0] – 2024-03-13\n\n### Added\n\n- Add support for Python 3.12.\n- Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n\n### Removed\n\n- Drop support for Python 3.7.\n\n### Deprecated\n\n- Deprecate `Exists` in favor of `Present` and `Absent` in\n  `assert_json_subset()`.\n\n## [0.12.0]\n\n### Added\n\n- Add `assert_not_regex()`.\n\n### Changed\n\n- Modernize the type stubs.\n\n### Removed\n\n- Drop support for Python 3.6.\n\n## [0.11.1]\n\n### Added\n\n- `assert_json_subset()` can now check for the existence or non-existence\n  of object members using the new `Exists` helper.\n- Non-string (or `Exists`) object member names in the first argument to\n  `assert_json_subset()` now raise a `TypeError`.\n\n## [0.11.0]\n\n### Removed\n\n- Drop support for Python 2.7 and 3.5.\n\n## [0.10.0]\n\n### Added\n\n- `AssertRaisesContext` and `AssertWarnsContext` now return themselves\n  when `__enter__()` is called. By extension it now easier to call\n  `add_test()` with `assert_raises()` et al:\n\n```python\nwith assert_raises(KeyError) as context:\n    context.add_test(...)\n    ...\n```\n\n- Add `AssertRaisesContext.exc_val` property to access the caught\n  exception after leaving the context manager:\n\n```python\nwith assert_raises(KeyError) as context:\n    ...\nassert_equal(\"expected message\", str(context.exc_val))\n```\n\n### Removed\n\n- Drop support for Python 3.4.\n\n## [0.9.1]\n\n### Changed\n\n- `AssertRaisesContext` and sub-classes are now generic over the\n  exception type.\n\n## [0.9.0]\n\n### Added\n\n- Add `assert_json_subset()`.\n\n## [0.8.6]\n\n### Added\n\n- Add support for Python 3.7 (contributed by Frank Niessink).\n\n## [0.8.5]\n\n### Added\n\n- Add `assert_dict_equal()`.\n- Add `assert_dict_superset()`.\n\n### Changed\n\n- `assert_equal()`: Use `assert_dict_equal()` if applicable.\n\n## [0.8.4]\n\n### Changed\n\n- `fail()` is now marked with `NoReturn` in type stub.\n\n### Fixed\n\n- Improve type annotations for Python 2.\n\n## [0.8.3]\n\n### Fixed\n\n- Fix type signature of `AssertRaisesContext.__exit__()`.\n\n## [0.8.2]\n\n### Added\n\n- Add a py.typed file to signal that this package supports type hints.\n\n## [0.8.1]\n\n### Fixed\n\n- `assert_raises_regex()`: Handle exceptions without any message correctly.\n\n## [0.8.0]\n\n### Added\n\n- assert_count_equal(): Add `msg_fmt` argument.\n- Add AssertRaisesErrnoContext, AssertRaisesRegexContext, and\n  AssertWarnsRegexContext.\n\n### Changed\n\n- Replace `msg` argument with `msg_fmt` in all assertions (except `fail()`).\n  This allows you to customize error messages more easily than before, because\n  `format()` with appropriate keyword arguments is now called on these\n  strings. See the documentation of individual assertions for the supported\n  arguments.\n- Replace AssertRaisesContext.msg and AssertWarnsContext.msg with msg_fmt.\n- assert_almost_equal(), assert_not_almost_equal(): Place msg_fmt as third\n  argument.\n\n## [0.7.3]\n\n### Added\n\n- Add assert_not_almost_equal().\n\n### Changed\n\n- assert_almost_equal(): Raise ValueError if diff <= 0.\n\n### Fixed\n\n- assert_almost_equal() would never fail if a delta was supplied and the\n  second number was smaller than the first.\n- Use fail() instead of raise AssertionError in a few assertions.\n\n## [0.7.2]\n\n### Added\n\n- Add assert_warns() and assert_warns_regex().\n\n## [0.7.1]\n\n### Changed\n\n- Distribute a wheel.\n- asserts is now a package, instead of a module.\n\n## [0.7.0]\n\n### Added\n\n- Add a stub file.\n- Add assert_count_equal().\n\n## [0.6]\n\n### Added\n\n- Add assert_less(), assert_less_equal(), assert_greater(), and\n  assert_greater_equal().\n- Add assert_not_is_instance().\n\n### Changed\n\n- assert_datetime_about_now()/assert_datetime_about_now_utc(): Handle\n  comparison with None more gracefully.\n\n## [0.5.1]\n\n### Added\n\n- Add the LICENSE file to the distribution.\n\n## [0.5]\n\nInitial release.\n"
    },
    {
      "path": "srittau_python-asserts/mypy.ini",
      "content": "[mypy]\ncheck_untyped_defs = True\ndisallow_subclassing_any = True\ndisallow_untyped_decorators = True\nno_implicit_optional = True\nno_implicit_reexport = True\nwarn_redundant_casts = True\nwarn_return_any = True\nwarn_unused_configs = True\nwarn_unused_ignores = True\n"
    },
    {
      "path": "srittau_python-asserts/poetry.lock",
      "content": "# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand.\n\n[[package]]\nname = \"mypy\"\nversion = \"1.10.0\"\ndescription = \"Optional static typing for Python\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_10_9_x86_64.whl\", hash = \"sha256:da1cbf08fb3b851ab3b9523a884c232774008267b1f83371ace57f412fe308c2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_11_0_arm64.whl\", hash = \"sha256:12b6bfc1b1a66095ab413160a6e520e1dc076a28f3e22f7fb25ba3b000b4ef99\"},\n    {file = \"mypy-1.10.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:9e36fb078cce9904c7989b9693e41cb9711e0600139ce3970c6ef814b6ebc2b2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-musllinux_1_1_x86_64.whl\", hash = \"sha256:2b0695d605ddcd3eb2f736cd8b4e388288c21e7de85001e9f85df9187f2b50f9\"},\n    {file = \"mypy-1.10.0-cp310-cp310-win_amd64.whl\", hash = \"sha256:cd777b780312ddb135bceb9bc8722a73ec95e042f911cc279e2ec3c667076051\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_10_9_x86_64.whl\", hash = \"sha256:3be66771aa5c97602f382230165b856c231d1277c511c9a8dd058be4784472e1\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_11_0_arm64.whl\", hash = \"sha256:8b2cbaca148d0754a54d44121b5825ae71868c7592a53b7292eeb0f3fdae95ee\"},\n    {file = \"mypy-1.10.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:1ec404a7cbe9fc0e92cb0e67f55ce0c025014e26d33e54d9e506a0f2d07fe5de\"},\n    {file = \"mypy-1.10.0-cp311-cp311-musllinux_1_1_x86_64.whl\", hash = \"sha256:e22e1527dc3d4aa94311d246b59e47f6455b8729f4968765ac1eacf9a4760bc7\"},\n    {file = \"mypy-1.10.0-cp311-cp311-win_amd64.whl\", hash = \"sha256:a87dbfa85971e8d59c9cc1fcf534efe664d8949e4c0b6b44e8ca548e746a8d53\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_10_9_x86_64.whl\", hash = \"sha256:a781f6ad4bab20eef8b65174a57e5203f4be627b46291f4589879bf4e257b97b\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_11_0_arm64.whl\", hash = \"sha256:b808e12113505b97d9023b0b5e0c0705a90571c6feefc6f215c1df9381256e30\"},\n    {file = \"mypy-1.10.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:8f55583b12156c399dce2df7d16f8a5095291354f1e839c252ec6c0611e86e2e\"},\n    {file = \"mypy-1.10.0-cp312-cp312-musllinux_1_1_x86_64.whl\", hash = \"sha256:4cf18f9d0efa1b16478c4c129eabec36148032575391095f73cae2e722fcf9d5\"},\n    {file = \"mypy-1.10.0-cp312-cp312-win_amd64.whl\", hash = \"sha256:bc6ac273b23c6b82da3bb25f4136c4fd42665f17f2cd850771cb600bdd2ebeda\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_10_9_x86_64.whl\", hash = \"sha256:9fd50226364cd2737351c79807775136b0abe084433b55b2e29181a4c3c878c0\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_11_0_arm64.whl\", hash = \"sha256:f90cff89eea89273727d8783fef5d4a934be2fdca11b47def50cf5d311aff727\"},\n    {file = \"mypy-1.10.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:fcfc70599efde5c67862a07a1aaf50e55bce629ace26bb19dc17cece5dd31ca4\"},\n    {file = \"mypy-1.10.0-cp38-cp38-musllinux_1_1_x86_64.whl\", hash = \"sha256:075cbf81f3e134eadaf247de187bd604748171d6b79736fa9b6c9685b4083061\"},\n    {file = \"mypy-1.10.0-cp38-cp38-win_amd64.whl\", hash = \"sha256:3f298531bca95ff615b6e9f2fc0333aae27fa48052903a0ac90215021cdcfa4f\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_10_9_x86_64.whl\", hash = \"sha256:fa7ef5244615a2523b56c034becde4e9e3f9b034854c93639adb667ec9ec2976\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_11_0_arm64.whl\", hash = \"sha256:3236a4c8f535a0631f85f5fcdffba71c7feeef76a6002fcba7c1a8e57c8be1ec\"},\n    {file = \"mypy-1.10.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:4a2b5cdbb5dd35aa08ea9114436e0d79aceb2f38e32c21684dcf8e24e1e92821\"},\n    {file = \"mypy-1.10.0-cp39-cp39-musllinux_1_1_x86_64.whl\", hash = \"sha256:92f93b21c0fe73dc00abf91022234c79d793318b8a96faac147cd579c1671746\"},\n    {file = \"mypy-1.10.0-cp39-cp39-win_amd64.whl\", hash = \"sha256:28d0e038361b45f099cc086d9dd99c15ff14d0188f44ac883010e172ce86c38a\"},\n    {file = \"mypy-1.10.0-py3-none-any.whl\", hash = \"sha256:f8c083976eb530019175aabadb60921e73b4f45736760826aa1689dda8208aee\"},\n    {file = \"mypy-1.10.0.tar.gz\", hash = \"sha256:3d087fcbec056c4ee34974da493a826ce316947485cef3901f511848e687c131\"},\n]\n\n[package.dependencies]\nmypy-extensions = \">=1.0.0\"\ntomli = {version = \">=1.1.0\", markers = \"python_version < \\\"3.11\\\"\"}\ntyping-extensions = \">=4.1.0\"\n\n[package.extras]\ndmypy = [\"psutil (>=4.0)\"]\ninstall-types = [\"pip\"]\nmypyc = [\"setuptools (>=50)\"]\nreports = [\"lxml\"]\n\n[[package]]\nname = \"mypy-extensions\"\nversion = \"1.0.0\"\ndescription = \"Type system extensions for programs checked with the mypy type checker.\"\noptional = false\npython-versions = \">=3.5\"\nfiles = [\n    {file = \"mypy_extensions-1.0.0-py3-none-any.whl\", hash = \"sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d\"},\n    {file = \"mypy_extensions-1.0.0.tar.gz\", hash = \"sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782\"},\n]\n\n[[package]]\nname = \"pastel\"\nversion = \"0.2.1\"\ndescription = \"Bring colors to your terminal.\"\noptional = false\npython-versions = \">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*\"\nfiles = [\n    {file = \"pastel-0.2.1-py2.py3-none-any.whl\", hash = \"sha256:4349225fcdf6c2bb34d483e523475de5bb04a5c10ef711263452cb37d7dd4364\"},\n    {file = \"pastel-0.2.1.tar.gz\", hash = \"sha256:e6581ac04e973cac858828c6202c1e1e81fee1dc7de7683f3e1ffe0bfd8a573d\"},\n]\n\n[[package]]\nname = \"poethepoet\"\nversion = \"0.26.1\"\ndescription = \"A task runner that works well with poetry.\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"poethepoet-0.26.1-py3-none-any.whl\", hash = \"sha256:aa43b443fec5d17d7e76771cccd484e5285805301721a74f059c483ad3276edd\"},\n    {file = \"poethepoet-0.26.1.tar.gz\", hash = \"sha256:aaad8541f6072617a60bcff2562d00779b58b353bd0f1847b06d8d0f2b6dc192\"},\n]\n\n[package.dependencies]\npastel = \">=0.2.1,<0.3.0\"\ntomli = \">=1.2.2\"\n\n[package.extras]\npoetry-plugin = [\"poetry (>=1.0,<2.0)\"]\n\n[[package]]\nname = \"ruff\"\nversion = \"0.4.8\"\ndescription = \"An extremely fast Python linter and code formatter, written in Rust.\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"ruff-0.4.8-py3-none-macosx_10_12_x86_64.whl\", hash = \"sha256:7663a6d78f6adb0eab270fa9cf1ff2d28618ca3a652b60f2a234d92b9ec89066\"},\n    {file = \"ruff-0.4.8-py3-none-macosx_11_0_arm64.whl\", hash = \"sha256:eeceb78da8afb6de0ddada93112869852d04f1cd0f6b80fe464fd4e35c330913\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl\", hash = \"sha256:aad360893e92486662ef3be0a339c5ca3c1b109e0134fcd37d534d4be9fb8de3\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl\", hash = \"sha256:284c2e3f3396fb05f5f803c9fffb53ebbe09a3ebe7dda2929ed8d73ded736deb\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl\", hash = \"sha256:a7354f921e3fbe04d2a62d46707e569f9315e1a613307f7311a935743c51a764\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl\", hash = \"sha256:72584676164e15a68a15778fd1b17c28a519e7a0622161eb2debdcdabdc71883\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl\", hash = \"sha256:9678d5c9b43315f323af2233a04d747409d1e3aa6789620083a82d1066a35199\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl\", hash = \"sha256:704977a658131651a22b5ebeb28b717ef42ac6ee3b11e91dc87b633b5d83142b\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:d05f8d6f0c3cce5026cecd83b7a143dcad503045857bc49662f736437380ad45\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_aarch64.whl\", hash = \"sha256:6ea874950daca5697309d976c9afba830d3bf0ed66887481d6bca1673fc5b66a\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_armv7l.whl\", hash = \"sha256:fc95aac2943ddf360376be9aa3107c8cf9640083940a8c5bd824be692d2216dc\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_i686.whl\", hash = \"sha256:384154a1c3f4bf537bac69f33720957ee49ac8d484bfc91720cc94172026ceed\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_x86_64.whl\", hash = \"sha256:e9d5ce97cacc99878aa0d084c626a15cd21e6b3d53fd6f9112b7fc485918e1fa\"},\n    {file = \"ruff-0.4.8-py3-none-win32.whl\", hash = \"sha256:6d795d7639212c2dfd01991259460101c22aabf420d9b943f153ab9d9706e6a9\"},\n    {file = \"ruff-0.4.8-py3-none-win_amd64.whl\", hash = \"sha256:e14a3a095d07560a9d6769a72f781d73259655919d9b396c650fc98a8157555d\"},\n    {file = \"ruff-0.4.8-py3-none-win_arm64.whl\", hash = \"sha256:14019a06dbe29b608f6b7cbcec300e3170a8d86efaddb7b23405cb7f7dcaf780\"},\n    {file = \"ruff-0.4.8.tar.gz\", hash = \"sha256:16d717b1d57b2e2fd68bd0bf80fb43931b79d05a7131aa477d66fc40fbd86268\"},\n]\n\n[[package]]\nname = \"tomli\"\nversion = \"2.0.1\"\ndescription = \"A lil' TOML parser\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"tomli-2.0.1-py3-none-any.whl\", hash = \"sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc\"},\n    {file = \"tomli-2.0.1.tar.gz\", hash = \"sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f\"},\n]\n\n[[package]]\nname = \"typing-extensions\"\nversion = \"4.12.2\"\ndescription = \"Backported and Experimental Type Hints for Python 3.8+\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"typing_extensions-4.12.2-py3-none-any.whl\", hash = \"sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d\"},\n    {file = \"typing_extensions-4.12.2.tar.gz\", hash = \"sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8\"},\n]\n\n[metadata]\nlock-version = \"2.0\"\npython-versions = \">=3.8.1\"\ncontent-hash = \"c010d5b718fed30569338ba631a7631751f5f6562c78491c16aa79927662e3f0\"\n"
    },
    {
      "path": "srittau_python-asserts/README.md",
      "content": "# Python Asserts\n\n[![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n[![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n[![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![GitHub Actions](https://img.shields.io/github/actions/workflow/status/srittau/python-asserts/test-and-lint.yml)](https://github.com/srittau/python-asserts/actions/workflows/test-and-lint.yml)\n\nStand-alone Assertions for Python\n\nThis package provides a few advantages over the assertions provided by\nunittest.TestCase:\n\n- Can be used stand-alone, for example:\n  - In test cases, not derived from TestCase.\n  - In fake and mock classes.\n  - In implementations as rich alternative to the assert statement.\n- PEP 8 compliance.\n- Custom stand-alone assertions can be written easily.\n- Arguably a better separation of concerns, since TestCase is responsible\n  for test running only, if assertion functions are used exclusively.\n\nThere are a few regressions compared to assertions from TestCase:\n\n- The default assertion class (`AssertionError`) can not be overwritten. This\n  is rarely a problem in practice.\n- asserts does not support the `addTypeEqualityFunc()` functionality.\n\nUsage:\n\n```python\n>>> from asserts import assert_true, assert_equal, assert_raises\n>>> my_var = 13\n>>> assert_equal(13, my_var)\n>>> assert_true(True, msg=\"custom failure message\")\n>>> with assert_raises(KeyError):\n...     raise KeyError()\n```\n\nFailure messages can be customized:\n\n```python\n>>> assert_equal(13, 14, msg_fmt=\"{got} is wrong, expected {expected}\")\nTraceback (most recent call last):\n  ...\nAssertionError: 14 is wrong, expected 13\n```\n"
    },
    {
      "path": "srittau_python-asserts/LICENSE",
      "content": "The MIT License (MIT)\n\nCopyright (c) 2014 Sebastian Rittau\n\nPermission is hereby granted, free of charge, to any person obtaining a copy of\nthis software and associated documentation files (the \"Software\"), to deal in\nthe Software without restriction, including without limitation the rights to\nuse, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of\nthe Software, and to permit persons to whom the Software is furnished to do so,\nsubject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS\nFOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR\nCOPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER\nIN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN\nCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.py",
      "content": "\"\"\"\nRich Assertions.\n\nThis module contains several rich standard assertions that can be used in unit\ntests and in implementations. Users are encouraged to define their own\nassertions, possibly using assertions from this package as a basis.\n\n    >>> assert_equal(13, 13)\n    >>> assert_equal(13, 14)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 13 != 14\n    >>> with assert_raises(KeyError):\n    ...     raise KeyError()\n    >>> with assert_raises(KeyError):\n    ...     pass\n    Traceback (most recent call last):\n        ...\n    AssertionError: KeyError not raised\n\n\"\"\"\n\nfrom __future__ import annotations\n\nimport re\nimport sys\nfrom datetime import datetime, timedelta, timezone\nfrom json import loads as json_loads\nfrom typing import Any, Callable, Set\nfrom warnings import WarningMessage, catch_warnings\n\nfrom typing_extensions import deprecated\n\n\ndef fail(msg=None):\n    \"\"\"Raise an AssertionError with the given message.\n\n    >>> fail(\"my message\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: my message\n\n    \"\"\"\n    raise AssertionError(msg or \"assertion failure\")\n\n\ndef assert_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is truthy.\n\n    >>> assert_true(\"Hello World!\")\n    >>> assert_true(\"\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: '' is not truthy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if not expr:\n        msg = \"{!r} is not truthy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is falsy.\n\n    >>> assert_false(\"\")\n    >>> assert_false(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not falsy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr:\n        msg = \"{!r} is not falsy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant True.\n\n    >>> assert_boolean_true(True)\n    >>> assert_boolean_true(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not True\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not True:\n        msg = \"{!r} is not True\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant False.\n\n    >>> assert_boolean_false(False)\n    >>> assert_boolean_false(0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 0 is not False\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not False:\n        msg = \"{!r} is not False\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is not None.\n\n    >>> assert_is_none(None)\n    >>> assert_is_none(False)\n    Traceback (most recent call last):\n        ...\n    AssertionError: False is not None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not None:\n        msg = \"{!r} is not None\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_not_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is None.\n\n    >>> assert_is_not_none(0)\n    >>> assert_is_not_none(None)\n    Traceback (most recent call last):\n        ...\n    AssertionError: expression is None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n    if expr is None:\n        msg = \"expression is None\"\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless first equals second, as determined by the '==' operator.\n\n    >>> assert_equal(5, 5.0)\n    >>> assert_equal(\"Hello World!\", \"Goodbye!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' != 'Goodbye!'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if isinstance(first, dict) and isinstance(second, dict):\n        assert_dict_equal(first, second, msg_fmt)\n    elif not first == second:\n        msg = \"{!r} != {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first equals second, as determined by the '==' operator.\n\n    >>> assert_not_equal(5, 8)\n    >>> assert_not_equal(-7, -7.0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: -7 == -7.0\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first == second:\n        msg = \"{!r} == {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are not equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_almost_equal(5, 5.00000001)\n    >>> assert_almost_equal(5, 5.001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 != 5.001 within 7 places\n    >>> assert_almost_equal(5, 5.001, places=2)\n    >>> assert_almost_equal(5, 5.001, delta=0.1)\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff < delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = not round(second - first, places)\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} != {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_not_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_not_almost_equal(5, 5.001)\n    >>> assert_not_almost_equal(5, 5.00000001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.00000001 within 7 places\n    >>> assert_not_almost_equal(5, 5.001, places=2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 within 2 places\n    >>> assert_not_almost_equal(5, 5.001, delta=0.1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 with delta=0.1\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff >= delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = bool(round(second - first, places))\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} == {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_dict_equal(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless first dictionary equals second.\n\n    The dictionaries are considered equal, if they both contain the same\n    keys, and their respective values are also equal.\n\n    >>> assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n    >>> assert_dict_equal({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n    * extra_keys - list of keys missing from left\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    extra_keys = list(second_keys - first_keys)\n    if missing_keys or extra_keys:\n        if missing_keys:\n            if len(missing_keys) == 1:\n                msg = \"key {!r} missing from right dict\".format(\n                    missing_keys[0]\n                )\n            else:\n                keys = \", \".join(sorted(repr(k) for k in missing_keys))\n                msg = \"keys {} missing from right dict\".format(keys)\n        else:\n            if len(extra_keys) == 1:\n                msg = \"extra key {!r} in right dict\".format(extra_keys[0])\n            else:\n                keys = \", \".join(sorted(repr(k) for k in extra_keys))\n                msg = \"extra keys {} in right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                missing_keys=missing_keys,\n                extra_keys=extra_keys,\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_dict_superset(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless second dictionary is a superset of the first.\n\n    The second dictionary must contain all keys of the first and their\n    values are equal (or a superset in case of dicts). But the second\n    dictionary can contain additional keys.\n\n    >>> assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n    >>> assert_dict_superset({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    if missing_keys:\n        if len(missing_keys) == 1:\n            msg = \"key {!r} missing from right dict\".format(missing_keys[0])\n        else:\n            keys = \", \".join(sorted(repr(k) for k in missing_keys))\n            msg = \"keys {} missing from right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg, first=first, second=second, missing_keys=missing_keys\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_less(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than second.\n\n    >>> assert_less('bar', 'foo')\n    >>> assert_less(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not less than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first < second:\n        msg = \"{!r} is not less than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_less_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than or equal to second.\n\n    >>> assert_less_equal('bar', 'foo')\n    >>> assert_less_equal(5, 5)\n    >>> assert_less_equal(6, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 6 is not less than or equal to 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first <= second:\n        msg = \"{!r} is not less than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than second.\n\n    >>> assert_greater('foo', 'bar')\n    >>> assert_greater(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first > second:\n        msg = \"{!r} is not greater than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than or equal to second.\n\n    >>> assert_greater_equal('foo', 'bar')\n    >>> assert_greater_equal(5, 5)\n    >>> assert_greater_equal(5, 6)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than or equal to 6\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first >= second:\n        msg = \"{!r} is not greater than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does not match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if not compiled.search(text):\n        msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_not_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if compiled.search(text):\n        msg = \"{!r} matches {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_is(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second do not refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is(list1, list1)\n    >>> assert_is(list1, list2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: [5, 'foo'] is not [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is not second:\n        msg = \"{!r} is not {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_is_not(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is_not(list1, list2)\n    >>> assert_is_not(list1, list1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: both arguments refer to [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is second:\n        msg = \"both arguments refer to {!r}\".format(first)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not in collection second.\n\n    >>> assert_in(\"foo\", [4, \"foo\", {}])\n    >>> assert_in(\"bar\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'bar' not in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n\n    if first not in second:\n        msg = \"{!r} not in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is in a collection second.\n\n    >>> assert_not_in(\"bar\", [4, \"foo\", {}])\n    >>> assert_not_in(\"foo\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n    if first in second:\n        msg = \"{!r} is in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_count_equal(sequence1, sequence2, msg_fmt=\"{msg}\"):\n    \"\"\"Compare the items of two sequences, ignoring order.\n\n    >>> assert_count_equal([1, 2], {2, 1})\n\n    Items missing in either sequence will be listed:\n\n    >>> assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\n\n    Items are counted in each sequence. This makes it useful to detect\n    duplicates:\n\n    >>> assert_count_equal({\"a\", \"b\"}, [\"a\", \"a\", \"b\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'a'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - first sequence\n    * second - second sequence\n    \"\"\"\n\n    def compare():\n        missing1 = list(sequence2)\n        missing2 = []\n        for item in sequence1:\n            try:\n                missing1.remove(item)\n            except ValueError:\n                missing2.append(item)\n        return missing1, missing2\n\n    def build_message():\n        msg = \"\"\n        if missing_from_1:\n            msg += \"missing from sequence 1: \" + \", \".join(\n                repr(i) for i in missing_from_1\n            )\n        if missing_from_1 and missing_from_2:\n            msg += \"; \"\n        if missing_from_2:\n            msg += \"missing from sequence 2: \" + \", \".join(\n                repr(i) for i in missing_from_2\n            )\n        return msg\n\n    missing_from_1, missing_from_2 = compare()\n    if missing_from_1 or missing_from_2:\n        fail(\n            msg_fmt.format(\n                msg=build_message(), first=sequence1, second=sequence2\n            )\n        )\n\n\ndef assert_between(lower_bound, upper_bound, expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an expression is not between certain bounds (inclusive).\n\n    >>> assert_between(5, 15, 5)\n    >>> assert_between(5, 15, 15)\n    >>> assert_between(5, 15, 4.9)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 4.9 is not between 5 and 15\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * lower - lower bound\n    * upper - upper bound\n    * expr - tested expression\n    \"\"\"\n\n    if not lower_bound <= expr <= upper_bound:\n        msg = \"{!r} is not between {} and {}\".format(\n            expr, lower_bound, upper_bound\n        )\n        fail(\n            msg_fmt.format(\n                msg=msg, lower=lower_bound, upper=upper_bound, expr=expr\n            )\n        )\n\n\ndef assert_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is not an instance of a class or tuple of classes.\n\n    >>> assert_is_instance(5, int)\n    >>> assert_is_instance('foo', (str, bytes))\n    >>> assert_is_instance(5, str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is an instance of <class 'int'>, expected <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if not isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}, expected {!r}\".format(\n            obj, obj.__class__, cls\n        )\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_not_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is an instance of a class or tuple of classes.\n\n    >>> assert_not_is_instance(5, str)\n    >>> assert_not_is_instance(5, (str, bytes))\n    >>> assert_not_is_instance('foo', str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is an instance of <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}\".format(obj, obj.__class__)\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_has_attr(obj, attribute, msg_fmt=\"{msg}\"):\n    \"\"\"Fail is an object does not have an attribute.\n\n    >>> assert_has_attr([], \"index\")\n    >>> assert_has_attr([], \"i_do_not_have_this\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: [] does not have attribute 'i_do_not_have_this'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * attribute - name of the attribute to check\n    \"\"\"\n\n    if not hasattr(obj, attribute):\n        msg = \"{!r} does not have attribute '{}'\".format(obj, attribute)\n        fail(msg_fmt.format(msg=msg, obj=obj, attribute=attribute))\n\n\n_EPSILON_SECONDS = 5\n\n\ndef assert_datetime_about_now(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of the local time.\n\n    >>> assert_datetime_about_now(datetime.now())\n    >>> assert_datetime_about_now(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now()\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\ndef assert_datetime_about_now_utc(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of UTC.\n\n    >>> assert_datetime_about_now_utc(datetime.now(timezone.utc).replace(tzinfo=None))\n    >>> assert_datetime_about_now_utc(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current UTC date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now(timezone.utc).replace(tzinfo=None)\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current UTC date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\nclass AssertRaisesContext:\n    \"\"\"A context manager to test for exceptions with certain properties.\n\n    When the context is left and no exception has been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertRaisesContext(TypeError)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: TypeError not raised\n\n    If an exception that is not a sub-class of the exception class provided\n    to the constructor is raised, it will be passed on:\n\n        >>> with context:\n        ...    raise ValueError(\"Wrong Class\")\n        Traceback (most recent call last):\n            ...\n        ValueError: Wrong Class\n\n    If the exception has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> def test(exc):\n        ...     assert_equal(\"Hello World!\", str(exc))\n        >>> context.add_test(test)\n        >>> with context:\n        ...     raise TypeError(\"Wrong Message\")\n        Traceback (most recent call last):\n            ...\n        AssertionError: 'Hello World!' != 'Wrong Message'\n\n    \"\"\"\n\n    def __init__(self, exception, msg_fmt=\"{msg}\"):\n        self.exception = exception\n        self.msg_fmt = msg_fmt\n        self._exc_type = exception\n        self._exc_val = None\n        self._exception_name = getattr(exception, \"__name__\", str(exception))\n        self._tests: list[Callable[[Any], object]] = []\n\n    def __enter__(self):\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        if not exc_type or not exc_val:\n            msg = \"{} not raised\".format(self._exception_name)\n            fail(self.format_message(msg))\n        self._exc_val = exc_val\n        if not issubclass(exc_type, self.exception):\n            return False\n        for test in self._tests:\n            test(exc_val)\n        return True\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n        )\n\n    def add_test(self, cb: Callable[[Any], object]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right exception\n        class was raised. The callback will get the raised exception as only\n        argument.\n\n        \"\"\"\n        self._tests.append(cb)\n\n    @property\n    def exc_val(self):\n        if self._exc_val is None:\n            raise RuntimeError(\"must be called after leaving the context\")\n        return self._exc_val\n\n\nclass AssertRaisesRegexContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions and their messages.\"\"\"\n\n    def __init__(self, exception, pattern, msg_fmt=\"{msg}\"):\n        super(AssertRaisesRegexContext, self).__init__(exception, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            pattern=self.pattern,\n            text=\"\",\n        )\n\n\nclass AssertRaisesErrnoContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions with errnos.\"\"\"\n\n    def __init__(self, exception, expected_errno, msg_fmt=\"{msg}\"):\n        super(AssertRaisesErrnoContext, self).__init__(exception, msg_fmt)\n        self.expected_errno = expected_errno\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            expected_errno=self.expected_errno,\n            actual_errno=None,\n        )\n\n\ndef assert_raises(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific exception is raised inside the context.\n\n    If a different type of exception is raised, it will not be caught.\n\n    >>> with assert_raises(TypeError):\n    ...     raise TypeError()\n    ...\n    >>> with assert_raises(TypeError):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: TypeError not raised\n    >>> with assert_raises(TypeError):\n    ...     raise ValueError(\"wrong error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    ValueError: wrong error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    \"\"\"\n\n    return AssertRaisesContext(exception, msg_fmt)\n\n\ndef assert_raises_regex(exception, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a message that matches a regular\n     expression is raised within the context.\n\n    The regular expression can be a regular expression string or object.\n\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Error #42\")\n    ...\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Generic Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Generic Error' does not match '\\\\\\\\d+'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * text - actual error text\n    * pattern - expected error message as regular expression string\n    \"\"\"\n\n    def test(exc):\n        compiled = re.compile(regex)\n        if not exc.args:\n            msg = \"{} without message\".format(exception.__name__)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=None,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n        text = exc.args[0]\n        if not compiled.search(text):\n            msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=text,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n\n    context = AssertRaisesRegexContext(exception, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\ndef assert_raises_errno(exception, errno, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a specific errno is raised with the\n     context.\n\n    >>> with assert_raises_errno(OSError, 42):\n    ...     raise OSError(42, \"OS Error\")\n    ...\n    >>> with assert_raises_errno(OSError, 44):\n    ...     raise OSError(17, \"OS Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: wrong errno: 44 != 17\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * expected_errno -\n    * actual_errno - raised errno or None if no matching exception was raised\n    \"\"\"\n\n    def check_errno(exc):\n        if errno != exc.errno:\n            msg = \"wrong errno: {!r} != {!r}\".format(errno, exc.errno)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                    expected_errno=errno,\n                    actual_errno=exc.errno,\n                )\n            )\n\n    context = AssertRaisesErrnoContext(exception, errno, msg_fmt)\n    context.add_test(check_errno)\n    return context\n\n\ndef assert_succeeds(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a specific exception is raised within the context.\n\n    This assertion should be used for cases, where successfully running a\n    function signals a successful test, and raising the exception of a\n    certain type signals a test failure. All other raised exceptions are\n    passed on and will usually still result in a test error. This can be\n    used to signal the intent of a block.\n\n    >>> l = [\"foo\", \"bar\"]\n    >>> with assert_succeeds(ValueError):\n    ...     i = l.index(\"foo\")\n    ...\n    >>> with assert_succeeds(ValueError):\n    ...     raise ValueError()\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: ValueError was unexpectedly raised\n    >>> with assert_succeeds(ValueError):\n    ...     raise TypeError(\"Wrong Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    TypeError: Wrong Error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    * exception - exception that was raised\n    \"\"\"\n\n    class _AssertSucceeds(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type and issubclass(exc_type, exception):\n                msg = exception.__name__ + \" was unexpectedly raised\"\n                fail(\n                    msg_fmt.format(\n                        msg=msg,\n                        exc_type=exception,\n                        exc_name=exception.__name__,\n                        exception=exc_val,\n                    )\n                )\n\n    return _AssertSucceeds()\n\n\nclass AssertWarnsContext(object):\n    \"\"\"A context manager to test for warnings with certain properties.\n\n    When the context is left and the expected warning has not been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertWarnsContext(DeprecationWarning)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    If the warning has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> from warnings import warn\n        >>> def test(warning):\n        ...     return False\n        >>> context.add_test(test)\n        >>> with context:\n        ...     warn(\"Wrong Message\", DeprecationWarning)\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    \"\"\"\n\n    def __init__(self, warning_class, msg_fmt=\"{msg}\"):\n        self._warning_class = warning_class\n        self._msg_fmt = msg_fmt\n        self._warning_context: catch_warnings[list[WarningMessage]] | None = (\n            None\n        )\n        self._warnings = []\n        self._tests: list[Callable[[Warning], bool]] = []\n\n    def __enter__(self):\n        self._warning_context = catch_warnings(record=True)\n        self._warnings = self._warning_context.__enter__()\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        assert self._warning_context is not None\n        self._warning_context.__exit__(exc_type, exc_val, exc_tb)\n        if not any(self._is_expected_warning(w) for w in self._warnings):\n            fail(self.format_message())\n\n    def format_message(self):\n        msg = \"{} not issued\".format(self._warning_class.__name__)\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n        )\n\n    def _is_expected_warning(self, warning) -> bool:\n        if not issubclass(warning.category, self._warning_class):\n            return False\n        return all(test(warning) for test in self._tests)\n\n    def add_test(self, cb: Callable[[Warning], bool]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right warning\n        class was issued. The callback will get the issued warning as only\n        argument and must return a boolean value.\n\n        \"\"\"\n        self._tests.append(cb)\n\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    \"\"\"A context manager to test for warnings and their messages.\"\"\"\n\n    def __init__(self, warning_class, pattern, msg_fmt=\"{msg}\"):\n        super(AssertWarnsRegexContext, self).__init__(warning_class, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self):\n        msg = \"no {} matching {} issued\".format(\n            self._warning_class.__name__, repr(self.pattern)\n        )\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n            pattern=self.pattern,\n        )\n\n\ndef assert_warns(warning_type, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific warning is issued inside the context.\n\n    If a different type of warning is issued, it will not be caught.\n\n    >>> from warnings import warn\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UserWarning)\n    ...\n    >>> with assert_warns(UserWarning):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UnicodeWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    \"\"\"\n    return AssertWarnsContext(warning_type, msg_fmt)\n\n\ndef assert_warns_regex(warning_type, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a warning with a message is issued inside the context.\n\n    The message can be a regular expression string or object.\n\n    >>> from warnings import warn\n    >>> with assert_warns_regex(UserWarning, r\"#\\\\d+\"):\n    ...     warn(\"Error #42\", UserWarning)\n    ...\n    >>> with assert_warns_regex(UserWarning, r\"Expected Error\"):\n    ...     warn(\"Generic Error\", UserWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: no UserWarning matching 'Expected Error' issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - warning type\n    * exc_name - warning type name\n    * pattern - expected warning message as regular expression string\n    \"\"\"\n\n    def test(warning):\n        return re.search(regex, str(warning.message)) is not None\n\n    context = AssertWarnsRegexContext(warning_type, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\nif sys.version_info >= (3,):\n    _Str = str\nelse:\n    _Str = unicode  # noqa: F821\n\n\ndef assert_json_subset(first, second):\n    \"\"\"Assert that a JSON object or array is a subset of another JSON object\n    or array.\n\n    The first JSON object or array must be supplied as a JSON-compatible\n    dict or list, the JSON object or array to check must be a string, an\n    UTF-8 bytes object, or a JSON-compatible list or dict.\n\n    A JSON non-object, non-array value is the subset of another JSON value,\n    if they are equal.\n\n    A JSON object is the subset of another JSON object if for each name/value\n    pair in the former there is a name/value pair in the latter with the same\n    name. Additionally, the value of the former pair must be a subset of the\n    value of the latter pair.\n\n    A JSON array is the subset of another JSON array, if they have the same\n    number of elements and each element in the former is a subset of the\n    corresponding element in the latter.\n\n    >>> assert_json_subset({}, '{}')\n    >>> assert_json_subset({}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({\"foo\": \"bar\"}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset([1, 2], '[1, 2]')\n    >>> assert_json_subset([2, 1], '[1, 2]')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element $[0] differs: 2 != 1\n    >>> assert_json_subset([{}], '[{\"foo\": \"bar\"}]')\n    >>> assert_json_subset({}, \"INVALID JSON\")\n    Traceback (most recent call last):\n    ...\n    json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)\n\n    In objects, the special classes `Present` and `Absent` can be used to\n    check for the presence or absence of a specific key:\n\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{}')\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{\"foo\": \"bar\"}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: spurious member 'foo' in object $\n    \"\"\"\n\n    if not isinstance(second, (dict, list, str, bytes)):\n        raise TypeError(\"second must be dict, list, str, or bytes\")\n    if isinstance(second, bytes):\n        second = second.decode(\"utf-8\")\n    if isinstance(second, _Str):\n        parsed_second = json_loads(second)\n    else:\n        parsed_second = second\n\n    if not isinstance(parsed_second, (dict, list)):\n        raise AssertionError(\n            \"second must decode to dict or list, not {}\".format(\n                type(parsed_second)\n            )\n        )\n\n    comparer = _JSONComparer(_JSONPath(\"$\"), first, parsed_second)\n    comparer.assert_()\n\n\nclass _JSONComparer:\n    def __init__(self, path, expected, actual):\n        self._path = path\n        self._expected = expected\n        self._actual = actual\n\n    def assert_(self):\n        self._assert_types_are_equal()\n        if isinstance(self._expected, dict):\n            self._assert_dicts_equal()\n        elif isinstance(self._expected, list):\n            self._assert_arrays_equal()\n        elif _is_present(self._expected):\n            pass\n        else:\n            self._assert_fundamental_values_equal()\n\n    def _assert_types_are_equal(self):\n        if self._types_differ():\n            self._raise_different_values()\n\n    def _types_differ(self):\n        if self._expected is None:\n            return self._actual is not None\n        elif isinstance(self._expected, (int, float)):\n            return not isinstance(self._actual, (int, float))\n        elif _is_present(self._expected):\n            return False\n        for type_ in [bool, str, _Str, list, dict]:\n            if isinstance(self._expected, type_):\n                return not isinstance(self._actual, type_)\n        else:\n            raise TypeError(\"unsupported type {}\".format(type(self._expected)))\n\n    def _assert_dicts_equal(self) -> None:\n        for name in self._expected:\n            if not isinstance(name, (str, Exists)):\n                raise TypeError(\n                    f\"{repr(name)} is not a valid object member name\",\n                )\n        self._assert_all_expected_keys_in_actual_dict()\n        self._assert_no_wrong_keys()\n        self._assert_dict_values_equal()\n\n    def _assert_all_expected_keys_in_actual_dict(self) -> None:\n        keys = self._expected_key_names.difference(self._actual.keys())\n        if keys:\n            self._raise_missing_element(keys)\n\n    def _assert_no_wrong_keys(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and _is_absent(self._expected[name]):\n                if name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name}' in object {{path}}\"\n                    )\n            if isinstance(name, Exists) and not self._expected[name]:\n                if name.member_name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name.member_name}' in object {{path}}\"\n                    )\n\n    def _assert_dict_values_equal(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and not _is_absent(self._expected[name]):\n                self._assert_json_value_equals_with_item(name)\n\n    @property\n    def _expected_key_names(self) -> Set[str]:\n        keys: Set[str] = set()\n        for k in self._expected.keys():\n            if isinstance(k, str):\n                if not _is_absent(self._expected[k]):\n                    keys.add(k)\n            elif isinstance(k, Exists) and self._expected[k]:\n                keys.add(k.member_name)\n        return keys\n\n    def _assert_arrays_equal(self):\n        if len(self._expected) != len(self._actual):\n            self._raise_different_sizes()\n        for i in range(len(self._expected)):\n            self._assert_json_value_equals_with_item(i)\n\n    def _assert_json_value_equals_with_item(self, item):\n        path = self._path.append(item)\n        expected = self._expected[item]\n        actual = self._actual[item]\n        _JSONComparer(path, expected, actual).assert_()\n\n    def _assert_fundamental_values_equal(self):\n        if self._expected != self._actual:\n            self._raise_different_values()\n\n    def _raise_different_values(self):\n        self._raise_assertion_error(\n            \"element {path} differs: {expected} != {actual}\"\n        )\n\n    def _raise_different_sizes(self):\n        self._raise_assertion_error(\n            \"JSON array {path} differs in size: \"\n            \"{expected_len} != {actual_len}\",\n            expected_len=len(self._expected),\n            actual_len=len(self._actual),\n        )\n\n    def _raise_missing_element(self, keys):\n        if len(keys) == 1:\n            format_string = \"element {elements} missing from element {path}\"\n            elements = repr(next(iter(keys)))\n        else:\n            format_string = \"elements {elements} missing from element {path}\"\n            sorted_keys = sorted(keys)\n            elements = (\n                \", \".join(repr(k) for k in sorted_keys[:-1])\n                + \", and \"\n                + repr(sorted_keys[-1])\n            )\n        self._raise_assertion_error(format_string, elements=elements)\n\n    def _raise_assertion_error(self, format_, **kwargs):\n        kwargs.update(\n            {\n                \"path\": self._path,\n                \"expected\": repr(self._expected),\n                \"actual\": repr(self._actual),\n            }\n        )\n        raise AssertionError(format_.format(**kwargs))\n\n\nclass _JSONPath:\n    def __init__(self, path):\n        self._path = path\n\n    def __str__(self):\n        return self._path\n\n    def append(self, item):\n        return _JSONPath(\"{0}[{1}]\".format(self._path, repr(item)))\n\n\nclass Present:\n    \"\"\"Helper class for presence checks in assert_json_subset().\"\"\"\n\n\ndef _is_present(o: object) -> bool:\n    return o is Present or isinstance(o, Present)\n\n\nclass Absent:\n    \"\"\"Helper class for absence checks in assert_json_subset().\"\"\"\n\n\ndef _is_absent(o: object) -> bool:\n    return o is Absent or isinstance(o, Absent)\n\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    \"\"\"Helper class for existence checks in assert_json_subset().\"\"\"\n\n    def __init__(self, member_name: str) -> None:\n        self.member_name = member_name\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.pyi",
      "content": "import datetime\nfrom collections.abc import Callable, Container, Iterable\nfrom contextlib import AbstractContextManager as ContextManager\nfrom re import Pattern\nfrom types import TracebackType\nfrom typing import Any, Generic, NoReturn, TypeVar\n\nfrom typing_extensions import deprecated\n\n_E = TypeVar(\"_E\", bound=BaseException)\n_S = TypeVar(\"_S\")\n\nclass AssertRaisesContext(Generic[_E]):\n    exception: type[_E]\n    msg_fmt: str\n    def __init__(self, exception: type[_E], msg_fmt: str = ...) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> bool: ...\n    def format_message(self, default_msg: str) -> str: ...\n    def add_test(self, cb: Callable[[_E], object]) -> None: ...\n    @property\n    def exc_val(self) -> _E: ...\n\nclass AssertRaisesErrnoContext(AssertRaisesContext[_E]):\n    expected_errno: int\n    def __init__(\n        self, exception: type[_E], expected_errno: int, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertRaisesRegexContext(AssertRaisesContext[_E]):\n    pattern: str\n    def __init__(\n        self, exception: type[_E], pattern: str, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertWarnsContext:\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> None: ...\n    def format_message(self) -> str: ...\n    def add_test(self, cb: Callable[[Warning], bool]) -> None: ...\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    pattern: str\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n\ndef fail(msg: str = ...) -> NoReturn: ...\ndef assert_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_not_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_not_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_dict_equal(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_dict_superset(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_less(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_less_equal(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater_equal(\n    first: Any, second: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_is(first: object, second: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_between(\n    lower_bound: Any, upper_bound: Any, expr: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_count_equal(\n    sequence1: Iterable[Any], sequence2: Iterable[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_has_attr(\n    obj: object, attribute: str, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now_utc(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_raises(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_raises_regex(\n    exception: type[BaseException],\n    regex: str | Pattern[str],\n    msg_fmt: str = ...,\n) -> AssertRaisesContext: ...\ndef assert_raises_errno(\n    exception: type[BaseException], errno: int, msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_succeeds(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> ContextManager: ...\ndef assert_warns(\n    warning_type: type[Warning], msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_warns_regex(\n    warning_type: type[Warning], regex: str, msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_json_subset(\n    first: dict[Any, Any] | list[Any],  # dict key can be 'str' or 'Exists'\n    second: dict[str, Any] | list[Any] | str | bytes,\n) -> None: ...\n\nclass Present: ...\nclass Absent: ...\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    member_name: str\n    def __init__(self, member_name: str) -> None: ...\n"
    },
    {
      "path": "srittau_python-asserts/asserts/py.typed",
      "content": ""
    }
  ],
  "BuggyCode": [
    {
      "path": "srittau_python-asserts/pyproject.toml",
      "content": "[tool.poetry]\nname = \"asserts\"\nversion = \"0.13.2.dev0\"\ndescription = \"Stand-alone Assertions\"\nreadme = \"README.md\"\nauthors = [\"Sebastian Rittau <srittau@rittau.biz>\"]\nlicense = \"MIT\"\nhomepage = \"https://github.com/srittau/python-asserts\"\nrepository = \"https://github.com/srittau/python-asserts\"\nclassifiers = [\n    \"Development Status :: 3 - Alpha\",\n    \"Intended Audience :: Developers\",\n    \"Topic :: Software Development :: Quality Assurance\",\n    \"Topic :: Software Development :: Testing\",\n]\npackages = [{ include = \"asserts\" }]\ninclude = [\"*/py.typed\", \"*.pyi\"]\n\n[tool.poetry.urls]\n\"GitHub\" = \"https://github.com/srittau/python-asserts\"\n\"Bug Tracker\" = \"https://github.com/srittau/python-asserts/issues\"\n\"Changes\" = \"https://github.com/srittau/python-asserts/blob/main/CHANGELOG.md\"\n\n[tool.poetry.dependencies]\npython = \">=3.8.1\"\ntyping-extensions = \"^4.10.0\"\n\n[tool.poetry.group.dev.dependencies]\nmypy = \">=1.9,<1.11\"\npoethepoet = \">=0.25,<0.27\"\nruff = \">=0.3.2,<0.5.0\"\n\n[tool.ruff]\nline-length = 79\ntarget-version = \"py38\"\n\n[tool.poe.tasks]\ntest = \"python3 -Wall -m unittest test_asserts\"\ndoctest = \"python3 -m doctest asserts/__init__.py\"\nlint = \"ruff check asserts test_asserts.py\"\ntypecheck = \"mypy asserts test_asserts.py\"\n\n[build-system]\nrequires = [\"poetry-core>=1.0.0\"]\nbuild-backend = \"poetry.core.masonry.api\"\n"
    },
    {
      "path": "srittau_python-asserts/test_asserts.py",
      "content": "# -*- coding: utf-8 -*-\n\nimport re\nimport sys\nfrom collections import OrderedDict\nfrom datetime import datetime, timedelta, timezone\nfrom json import JSONDecodeError\nfrom unittest import TestCase\nfrom warnings import catch_warnings, simplefilter, warn\n\nfrom asserts import (\n    Absent,\n    Exists,\n    Present,\n    assert_almost_equal,\n    assert_between,\n    assert_boolean_false,\n    assert_boolean_true,\n    assert_count_equal,\n    assert_datetime_about_now,\n    assert_datetime_about_now_utc,\n    assert_dict_equal,\n    assert_dict_superset,\n    assert_equal,\n    assert_false,\n    assert_greater,\n    assert_greater_equal,\n    assert_has_attr,\n    assert_in,\n    assert_is,\n    assert_is_instance,\n    assert_is_none,\n    assert_is_not,\n    assert_is_not_none,\n    assert_json_subset,\n    assert_less,\n    assert_less_equal,\n    assert_not_almost_equal,\n    assert_not_equal,\n    assert_not_in,\n    assert_not_is_instance,\n    assert_not_regex,\n    assert_raises,\n    assert_raises_errno,\n    assert_raises_regex,\n    assert_regex,\n    assert_succeeds,\n    assert_true,\n    assert_warns,\n    assert_warns_regex,\n    fail,\n)\n\n\nclass Box:\n    def __init__(self, initial_value):\n        self.value = initial_value\n\n\nclass _DummyObject(object):\n    def __init__(self, value=\"x\"):\n        self.value = value\n\n    def __repr__(self):\n        return \"<Dummy>\"\n\n\ndef _assert_raises_assertion(expected_message):\n    \"\"\"Fail if the context does not raise an AssertionError or the exception\n    message does not match.\n\n    This is used to test assertions, without using those assertions.\n\n    \"\"\"\n\n    class Context(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type is None:\n                raise AssertionError(\"no AssertionError raised\")\n            if exc_type != AssertionError:\n                return False\n            if str(exc_val) != expected_message:\n                raise AssertionError(\n                    \"expected exception message {!r}, got {!r}\".format(\n                        expected_message, str(exc_val)\n                    )\n                )\n            return True\n\n    return Context()\n\n\nclass AssertTest(TestCase):\n    _type_string = \"type\" if sys.version_info[0] < 3 else \"class\"\n\n    # fail()\n\n    def test_fail__default_message(self):\n        with _assert_raises_assertion(\"assertion failure\"):\n            fail()\n\n    def test_fail__with_message(self):\n        with _assert_raises_assertion(\"test message\"):\n            fail(\"test message\")\n\n    # assert_true()\n\n    def test_assert_true__truthy_value(self):\n        assert_true(\"Hello World!\")\n\n    def test_assert_true__falsy_value__default_message(self):\n        with _assert_raises_assertion(\"'' is not truthy\"):\n            assert_true(\"\")\n\n    def test_assert_true__falsy_value__custom_message(self):\n        with _assert_raises_assertion(\"0 is not truthy;0\"):\n            assert_true(0, \"{msg};{expr}\")\n\n    # assert_false()\n\n    def test_assert_false__falsy_value(self):\n        assert_false(\"\")\n\n    def test_assert_false__truthy_value__default_message(self):\n        with _assert_raises_assertion(\"25 is not falsy\"):\n            assert_false(25)\n\n    def test_assert_false__truthy_value__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is not falsy;foo\"):\n            assert_false(\"foo\", \"{msg};{expr}\")\n\n    # assert_boolean_true()\n\n    def test_assert_boolean_true__true(self):\n        assert_boolean_true(True)\n\n    def test_assert_boolean_true__false__custom_message(self):\n        with _assert_raises_assertion(\"'Foo' is not True;Foo\"):\n            assert_boolean_true(\"Foo\", \"{msg};{expr}\")\n\n    def test_assert_boolean_true__truthy__default_message(self):\n        with _assert_raises_assertion(\"1 is not True\"):\n            assert_boolean_true(1)\n\n    # assert_boolean_false()\n\n    def test_assert_boolean_false__false(self):\n        assert_boolean_false(False)\n\n    def test_assert_boolean_false__true__default_message(self):\n        with _assert_raises_assertion(\"'foo' is not False\"):\n            assert_boolean_false(\"foo\")\n\n    def test_assert_boolean_false__falsy__custom_message(self):\n        with _assert_raises_assertion(\"0 is not False;0\"):\n            assert_boolean_false(0, \"{msg};{expr}\")\n\n    # assert_is_none()\n\n    def test_assert_is_none__none(self):\n        assert_is_none(None)\n\n    def test_assert_is_none__string__default_message(self):\n        with _assert_raises_assertion(\"'' is not None\"):\n            assert_is_none(\"\")\n\n    def test_assert_is_none__int__custom_message(self):\n        with _assert_raises_assertion(\"55 is not None;55\"):\n            assert_is_none(55, \"{msg};{expr}\")\n\n    # assert_is_not_none()\n\n    def test_assert_is_not_none__string(self):\n        assert_is_not_none(\"\")\n\n    def test_assert_is_not_none__none__default_message(self):\n        with _assert_raises_assertion(\"expression is None\"):\n            assert_is_not_none(None)\n\n    def test_assert_is_not_none__none__custom_message(self):\n        with _assert_raises_assertion(\"expression is None;None\"):\n            assert_is_not_none(None, \"{msg};{expr!r}\")\n\n    # assert_equal()\n\n    def test_assert_equal__equal_strings(self):\n        assert_equal(\"foo\", \"foo\")\n\n    def test_assert_equal__equal_objects(self):\n        class MyClass(object):\n            def __eq__(self, other):\n                return True\n\n        assert_equal(MyClass(), MyClass())\n\n    def test_assert_equal__not_equal__default_message(self):\n        with _assert_raises_assertion(\"'string' != 55\"):\n            assert_equal(\"string\", 55)\n\n    def test_assert_equal__not_equal__custom_message(self):\n        with _assert_raises_assertion(\"'string' != 55;'string';55\"):\n            assert_equal(\"string\", 55, \"{msg};{first!r};{second!r}\")\n\n    def test_assert_equal__dict(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_equal({\"foo\": 5}, {})\n\n    # assert_not_equal()\n\n    def test_assert_not_equal__not_equal(self):\n        assert_not_equal(\"abc\", \"def\")\n\n    def test_assert_not_equal__equal__default_message(self):\n        with _assert_raises_assertion(\"'abc' == 'abc'\"):\n            assert_not_equal(\"abc\", \"abc\")\n\n    def test_assert_not_equal__equal__custom_message(self):\n        with _assert_raises_assertion(\"1.0 == 1;1.0;1\"):\n            assert_not_equal(1.0, 1, \"{msg};{first};{second}\")\n\n    # assert_almost_equal()\n\n    def test_assert_almost_equal__same(self):\n        assert_almost_equal(5, 5)\n\n    def test_assert_almost_equal__similar__defaults(self):\n        assert_almost_equal(5, 5.00000001)\n\n    def test_assert_almost_equal__similar__places(self):\n        assert_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_almost_equal__similar__delta(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__similar__delta_reverse(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__default_message(self):\n        with _assert_raises_assertion(\"5 != 5.0001 within 7 places\"):\n            assert_almost_equal(5, 5.0001)\n\n    def test_assert_almost_equal__not_similar__places__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 within 3 places\"):\n            assert_almost_equal(5, 6, places=3)\n\n    def test_assert_almost_equal__not_similar__delta__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1\"):\n            assert_almost_equal(5, 6, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__delta_reverse(self):\n        with _assert_raises_assertion(\"6 != 5 with delta=0.3\"):\n            assert_almost_equal(6, 5, delta=0.3)\n\n    def test_assert_almost_equal__not_similar__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 7 places;5;-5;7;None\"):\n            assert_almost_equal(\n                5, -5, msg_fmt=\"{msg};{first};{second};{places};{delta!r}\"\n            )\n\n    def test_assert_almost_equal__not_similar__places__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 3 places;5;-5;3;None\"):\n            assert_almost_equal(\n                5,\n                -5,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_almost_equal__not_similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1;5;6;None;0.1\"):\n            assert_almost_equal(\n                5,\n                6,\n                delta=0.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_almost_equal__wrong_types(self):\n        try:\n            assert_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__places_and_delta(self):\n        try:\n            assert_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_not_almost_equal()\n\n    def test_assert_not_almost_equal__same(self):\n        with _assert_raises_assertion(\"5 == 5 within 7 places\"):\n            assert_not_almost_equal(5, 5)\n\n    def test_assert_not_almost_equal__similar__defaults(self):\n        with _assert_raises_assertion(\"5 == 5.00000001 within 7 places\"):\n            assert_not_almost_equal(5, 5.00000001)\n\n    def test_assert_not_almost_equal__similar__places(self):\n        with _assert_raises_assertion(\"5 == 5.0001 within 3 places\"):\n            assert_not_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_not_almost_equal__similar__delta(self):\n        with _assert_raises_assertion(\"5 == 5.1 with delta=0.1\"):\n            assert_not_almost_equal(5, 5.1, delta=0.1)\n\n    def test_assert_not_almost_equal__similar__delta_reverse(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.3\"):\n            assert_almost_equal(5, 6, delta=0.3)\n\n    def test_assert_not_almost_equal__not_similar(self):\n        assert_not_almost_equal(5, 5.0001)\n\n    def test_assert_not_almost_equal__not_similar__delta(self):\n        assert_not_almost_equal(5, 5.1, delta=0.05)\n\n    def test_assert_not_almost_equal__not_similar__delta_reverse(self):\n        assert_not_almost_equal(5.1, 5, delta=0.05)\n\n    def test_assert_not_almost_equal__similar__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.00000001 within 7 places;5;5.00000001;7;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.00000001,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__places__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.0001 within 3 places;5;5.0001;3;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.0001,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 == 6 with delta=1.1;5;6;None;1.1\"):\n            assert_not_almost_equal(\n                5,\n                6,\n                delta=1.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_not_almost_equal__wrong_types(self):\n        try:\n            assert_not_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_not_almost_equal__places_and_delta(self):\n        try:\n            assert_not_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_not_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_not_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_dict_equal()\n\n    def test_assert_dict_equal__empty_dicts(self):\n        assert_dict_equal({}, {})\n\n    def test_assert_dict_equal__dicts_are_equal(self):\n        assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_equal__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_equal({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_equal__one_key_missing_from_left(self):\n        with _assert_raises_assertion(\"extra key 'foo' in right dict\"):\n            assert_dict_equal({\"bar\": 10}, {\"bar\": 10, \"foo\": 5})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_left(self):\n        with _assert_raises_assertion(\"extra keys 'bar', 'foo' in right dict\"):\n            assert_dict_equal({\"baz\": 15}, {\"foo\": 5, \"bar\": 10, \"baz\": 15})\n\n    def test_assert_dict_equal__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_equal({\"foo\": 15}, {\"foo\": 10})\n\n    def test_assert_dict_equal__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_equal({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_equal({5: \"\", \"foo\": \"\"}, {})\n        with _assert_raises_assertion(\"extra key 10 in right dict\"):\n            assert_dict_equal({}, {10: \"foo\"})\n        with _assert_raises_assertion(\"extra keys 'foo', 5 in right dict\"):\n            assert_dict_equal({}, {5: \"\", \"foo\": \"\"})\n\n    def test_assert_dict_equal__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal(\n                {\"foo\": \"\", \"bar\": \"\", \"baz\": 5},\n                {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"},\n            )\n        with _assert_raises_assertion(\"extra key 'extra' in right dict\"):\n            assert_dict_equal(\n                {\"bar\": \"\", \"baz\": 5}, {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"}\n            )\n\n    def test_assert_dict_equal__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo'];['bar']\"\n        ):\n            assert_dict_equal(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{missing_keys!r};{extra_keys!r}\",\n            )\n\n    def test_assert_dict_equal__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_equal(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_dict_superset()\n\n    def test_assert_dict_superset__empty_dicts(self):\n        assert_dict_superset({}, {})\n\n    def test_assert_dict_superset__dicts_are_equal(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_superset__dicts_is_superset(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n\n    def test_assert_dict_superset__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_superset__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_superset({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_superset__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_superset({\"foo\": 15}, {\"foo\": 10, \"bar\": 15})\n\n    def test_assert_dict_superset__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_superset({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_superset({5: \"\", \"foo\": \"\"}, {})\n\n    def test_assert_dict_superset__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"foo\": \"\", \"bar\": 5}, {\"bar\": 1})\n\n    def test_assert_dict_superset__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo']\"\n        ):\n            assert_dict_superset(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\" \"{missing_keys!r}\",\n            )\n\n    def test_assert_dict_superset__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_superset(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_less()\n\n    def test_assert_less(self):\n        assert_less(4, 5)\n        with _assert_raises_assertion(\"5 is not less than 5\"):\n            assert_less(5, 5)\n        with _assert_raises_assertion(\"'foo' is not less than 'bar'\"):\n            assert_less(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than 5;6;5\"):\n            assert_less(6, 5, \"{msg};{first};{second}\")\n\n    # assert_less_equal()\n\n    def test_assert_less_equal(self):\n        assert_less_equal(4, 5)\n        assert_less_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'foo' is not less than or equal to 'bar'\"\n        ):\n            assert_less_equal(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than or equal to 5;6;5\"):\n            assert_less_equal(6, 5, \"{msg};{first};{second}\")\n\n    # assert_greater()\n\n    def test_assert_greater(self):\n        assert_greater(5, 4)\n        with _assert_raises_assertion(\"5 is not greater than 5\"):\n            assert_greater(5, 5)\n        with _assert_raises_assertion(\"'bar' is not greater than 'foo'\"):\n            assert_greater(\"bar\", \"foo\")\n        with _assert_raises_assertion(\"5 is not greater than 6;5;6\"):\n            assert_greater(5, 6, \"{msg};{first};{second}\")\n\n    # assert_greater_equal()\n\n    def test_assert_greater_equal(self):\n        assert_greater_equal(5, 4)\n        assert_greater_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'bar' is not greater than or equal to 'foo'\"\n        ):\n            assert_greater_equal(\"bar\", \"foo\")\n        with _assert_raises_assertion(\n            \"5 is not greater than or equal to 6;5;6\"\n        ):\n            assert_greater_equal(5, 6, \"{msg};{first};{second}\")\n\n    # assert_regex()\n\n    def test_assert_regex__matches_string(self):\n        assert_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_regex__matches_regex(self):\n        regex = re.compile(\"is.*test\")\n        assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", \"not found\")\n\n    def test_assert_regex__does_not_match_regex__default_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';\"\n            \"'Wrong text';'not found'\"\n        ):\n            assert_regex(\n                \"Wrong text\", r\"not found\", \"{msg};{text!r};{pattern!r}\"\n            )\n\n    def test_assert_regex__does_not_match_regex__custom_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';'Wrong text';\"\n            \"'not found'\"\n        ):\n            assert_regex(\"Wrong text\", regex, \"{msg};{text!r};{pattern!r}\")\n\n    # assert_not_regex()\n\n    def test_assert_not_regex__does_not_match_string(self):\n        assert_not_regex(\"This is a test text\", \"no match\")\n\n    def test_assert_not_regex__does_not_match_regex(self):\n        regex = re.compile(\"no match\")\n        assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_not_regex__matches_regex__default_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';\"\n            \"'This is a test text';'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\",\n                \"is.*test\",\n                \"{msg};{text!r};{pattern!r}\",\n            )\n\n    def test_assert_not_regex__matches_regex__custom_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';'This is a test text';\"\n            \"'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\", regex, \"{msg};{text!r};{pattern!r}\"\n            )\n\n    # assert_is()\n\n    def test_assert_is__same(self):\n        x = _DummyObject()\n        assert_is(x, x)\n\n    def test_assert_is__not_same__default_message(self):\n        with _assert_raises_assertion(\"'x' is not 'y'\"):\n            assert_is(\"x\", \"y\")\n\n    def test_assert_is__equal_but_not_same__custom_message(self):\n        x = \"x\"\n        y = _DummyObject(\"y\")\n        with _assert_raises_assertion(\"'x' is not <Dummy>;'x';y\"):\n            assert_is(x, y, \"{msg};{first!r};{second.value}\")\n\n    # assert_is_not()\n\n    def test_assert_is_not__not_same(self):\n        x = _DummyObject()\n        y = _DummyObject()\n        assert_is_not(x, y)\n\n    def test_assert_is_not__same__default_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>\"):\n            assert_is_not(x, x)\n\n    def test_assert_is_not__same__custom_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>;x;x\"):\n            assert_is_not(x, x, \"{msg};{first.value};{second.value}\")\n\n    # assert_in()\n\n    def test_assert_in__contains(self):\n        assert_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_in__does_not_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' not in []\"):\n            assert_in(\"foo\", [])\n\n    def test_assert_in__does_not_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' not in [];'foo';[]\"):\n            assert_in(\"foo\", [], \"{msg};{first!r};{second!r}\")\n\n    # assert_not_in()\n\n    def test_assert_not_in__does_not_contain(self):\n        assert_not_in(\"foo\", [])\n\n    def test_assert_not_in__does_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar', 'baz']\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_not_in__does_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar'];'foo';bar\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\"], \"{msg};{first!r};{second[1]}\")\n\n    # assert_count_equal()\n\n    def test_assert_count_equal__equal(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], [\"a\"])\n\n    def test_assert_count_equal__equal_differing_types(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], {\"a\"})\n\n    def test_assert_count_equal__ignore_order(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\", \"b\"], [\"b\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal([], {\"a\"})\n\n    def test_assert_count_equal__multiple_missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'b', 'c'\"):\n            assert_count_equal([\"a\"], [\"a\", \"b\", \"c\"])\n\n    def test_assert_count_equal__respect_duplicates(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal({\"a\"}, [\"a\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence2(self):\n        with _assert_raises_assertion(\"missing from sequence 2: 'a', 'c'\"):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"b\"])\n\n    def test_assert_count_equal__missing_from_both(self):\n        msg = \"missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\"\n        with _assert_raises_assertion(msg):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n\n    def test_assert_count_equal__custom_message(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a';[];['a']\"):\n            assert_count_equal([], [\"a\"], \"{msg};{first};{second}\")\n\n    # assert_between()\n\n    def test_assert_between__within_range(self):\n        assert_between(0, 10, 0)\n        assert_between(0, 10, 10)\n        assert_between(0, 10, 5)\n\n    def test_assert_between__too_low__default_message(self):\n        with _assert_raises_assertion(\"-1 is not between 0 and 10\"):\n            assert_between(0, 10, -1)\n\n    def test_assert_between__too_high__custom_message(self):\n        with _assert_raises_assertion(\"11 is not between 0 and 10;0;10;11\"):\n            assert_between(0, 10, 11, \"{msg};{lower};{upper};{expr}\")\n\n    # assert_is_instance()\n\n    def _is_instance_message(self, expr, expected_type, real_type):\n        expected_message = (\n            \"{!r} is an instance of <class {}>, expected {}\".format(\n                expr, real_type, expected_type\n            )\n        )\n        if sys.version_info[0] < 3:\n            return expected_message.replace(\"class\", \"type\")\n        else:\n            return expected_message\n\n    def test_assert_is_instance__single_type(self):\n        assert_is_instance(4, int)\n        assert_is_instance(OSError(), Exception)\n\n    def test_assert_is_instance__multiple_types(self):\n        assert_is_instance(4, (str, int))\n\n    def test_assert_is_instance__default_message(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        with _assert_raises_assertion(expected_message):\n            assert_is_instance(\"my string\", int)\n\n    def test_assert_is_instance__custom_message_single_type(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>,)\".format(expected_message)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\"my string\", int, \"{msg};{obj};{types}\")\n\n    def test_assert_is_instance__custom_message_multiple_types(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"(<class 'int'>, <class 'float'>)\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>, <class 'float'>)\".format(\n            expected_message\n        )\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\n                \"my string\", (int, float), \"{msg};{obj};{types}\"\n            )\n\n    # assert_not_is_instance()\n\n    def _not_is_instance_message(self, obj):\n        expected_message = \"{!r} is an instance of {}\".format(\n            obj, obj.__class__\n        )\n        if sys.version_info[0] < 3:\n            expected_message = expected_message.replace(\"class\", \"type\")\n            expected_message = expected_message.replace(\n                \"type 'OSError'\", \"type 'exceptions.OSError'\"\n            )\n        return expected_message\n\n    def test_assert_not_is_instance__single_type(self):\n        assert_not_is_instance(4, str)\n\n    def test_assert_not_is_instance__multiple_types(self):\n        assert_not_is_instance(4, (str, bytes))\n\n    def test_assert_not_is_instance__default_message(self):\n        obj = OSError()\n        expected_message = self._not_is_instance_message(obj)\n        with _assert_raises_assertion(expected_message):\n            assert_not_is_instance(obj, Exception)\n\n    def test_assert_not_is_instance__custom_message__single_type(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>,)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", str, \"{msg};{obj};{types!r}\")\n\n    def test_assert_not_is_instance__custom_message__multiple_types(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>, <class 'int'>)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", (str, int), \"{msg};{obj};{types!r}\")\n\n    # assert_has_attr()\n\n    def test_assert_has_attr__has_attribute(self):\n        d = _DummyObject()\n        assert_has_attr(d, \"value\")\n\n    def test_assert_has_attr__does_not_have_attribute__default_message(self):\n        d = _DummyObject()\n        with _assert_raises_assertion(\"<Dummy> does not have attribute 'foo'\"):\n            assert_has_attr(d, \"foo\")\n\n    def test_assert_has_attr__does_not_have_attribute__custom_message(self):\n        d = _DummyObject()\n        expected = \"<Dummy> does not have attribute 'foo';<Dummy>;foo\"\n        with _assert_raises_assertion(expected):\n            assert_has_attr(d, \"foo\", msg_fmt=\"{msg};{obj!r};{attribute}\")\n\n    # assert_datetime_about_now()\n\n    def test_assert_datetime_about_now__close(self):\n        assert_datetime_about_now(datetime.now())\n\n    def test_assert_datetime_about_now__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(None)\n\n    def test_assert_datetime_about_now__none__custom_message(self):\n        dt = datetime.now().date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now__too_low(self):\n        then = datetime.now() - timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__too_high(self):\n        then = datetime.now() + timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"^datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            \"is not close to current date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now().date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_datetime_about_now_utc()\n\n    def test_assert_datetime_about_now_utc__close(self):\n        assert_datetime_about_now_utc(\n            datetime.now(timezone.utc).replace(tzinfo=None)\n        )\n\n    def test_assert_datetime_about_now_utc__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(None)\n\n    def test_assert_datetime_about_now_utc__none__custom_message(self):\n        dt = datetime.now(timezone.utc).date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now_utc__too_low(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) - timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__too_high(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) + timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            r\"is not close to current UTC date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now(timezone.utc).date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current UTC date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_raises()\n\n    def test_assert_raises__raises_right_exception(self):\n        with assert_raises(KeyError):\n            raise KeyError()\n\n    def test_assert_raises__exc_val(self):\n        exc = KeyError()\n        with assert_raises(KeyError) as context:\n            raise exc\n        assert_is(exc, context.exc_val)\n\n    def test_assert_raises__exc_val_within_context(self):\n        with assert_raises(RuntimeError):\n            with assert_raises(KeyError) as context:\n                context.exc_val\n\n    def test_assert_raises__raises_subclass(self):\n        class MyError(IndexError):\n            pass\n\n        with assert_raises(IndexError):\n            raise MyError()\n\n    def test_assert_raises__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises(KeyError):\n                pass\n\n    def test_assert_raises__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError\"\n        with _assert_raises_assertion(expected):\n            with assert_raises(\n                KeyError, msg_fmt=\"{msg};{exc_type.__name__};{exc_name}\"\n            ):\n                pass\n\n    def test_assert_raises__wrong_exception_raised(self):\n        try:\n            with assert_raises(IndexError):\n                raise KeyError()\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(exc):\n            assert_is_instance(exc, KeyError)\n            called.value = True\n\n        with assert_raises(KeyError) as context:\n            context.add_test(extra_test)\n            raise KeyError()\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_raises__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_):\n            called.value = True\n\n        with assert_raises(AssertionError):\n            with assert_raises(KeyError) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_raises_regex()\n\n    def test_assert_raises_regex__raises_right_exception(self):\n        with assert_raises_regex(KeyError, r\"test.*\"):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__raises_right_exception__compiled(self):\n        with assert_raises_regex(KeyError, re.compile(r\"test.*\")):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                pass\n\n    def test_assert_raises_regex__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError;'';test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_regex__no_message__default_message(self):\n        with _assert_raises_assertion(\"KeyError without message\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError()\n\n    def test_assert_raises_regex__no_message__custom_message(self):\n        expected = \"KeyError without message;KeyError;KeyError;None;test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError()\n\n    def test_assert_raises_regex__wrong_exception_raised(self):\n        try:\n            with assert_raises_regex(IndexError, \"test message\"):\n                raise KeyError(\"test message\")\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises_regex__wrong_error__default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__pattern_default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, re.compile(r\"test\")):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__custom_message(self):\n        expected = (\n            \"'wrong message' does not match 'test';KeyError;KeyError;\"\n            \"'wrong message';test\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError(\"wrong message\")\n\n    # assert_raises_errno()\n\n    def test_assert_raises_errno__right_errno(self):\n        with assert_raises_errno(OSError, 20):\n            raise OSError(20, \"Test error\")\n\n    def test_assert_raises_errno__no_exception_raised__default_message(self):\n        with _assert_raises_assertion(\"OSError not raised\"):\n            with assert_raises_errno(OSError, 20):\n                pass\n\n    def test_assert_raises_errno__no_exception_raised__custom_message(self):\n        expected = \"OSError not raised;OSError;OSError;20;None\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_errno__wrong_class_raised(self):\n        class RightClass(OSError):\n            pass\n\n        class WrongClass(OSError):\n            pass\n\n        try:\n            with assert_raises_errno(RightClass, 20):\n                raise WrongClass(20, \"Test error\")\n        except WrongClass:\n            pass\n        else:\n            raise AssertionError(\"WrongClass was not raised\")\n\n    def test_assert_raises_errno__wrong_errno__default_message(self):\n        with _assert_raises_assertion(\"wrong errno: 20 != 1\"):\n            with assert_raises_errno(OSError, 20):\n                raise OSError(1, \"Test error\")\n\n    def test_assert_raises_errno__wrong_errno__custom_message(self):\n        expected = \"wrong errno: 20 != 1;OSError;OSError;20;1\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                raise OSError(1, \"Test error\")\n\n    # assert_succeeds()\n\n    def test_assert_succeeds__no_exception_raised(self):\n        with assert_succeeds(KeyError):\n            pass\n\n    def test_assert_succeeds__expected_exception__default_message(self):\n        with _assert_raises_assertion(\"KeyError was unexpectedly raised\"):\n            with assert_succeeds(KeyError):\n                raise KeyError()\n\n    def test_assert_succeeds__expected_exception__custom_message(self):\n        expected = (\n            \"KeyError was unexpectedly raised;KeyError;KeyError;test error\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{exception.args[0]}\"\n            )\n            with assert_succeeds(KeyError, msg_fmt=msg_fmt):\n                raise KeyError(\"test error\")\n\n    def test_assert_succeeds__unexpected_exception(self):\n        try:\n            with assert_succeeds(ValueError):\n                raise KeyError()\n        except KeyError:\n            pass\n        else:\n            raise AssertionError(\"KeyError was not raised\")\n\n    # assert_warns()\n\n    def test_assert_warns__default_message(self):\n        with assert_raises_regex(AssertionError, r\"^ImportWarning not issued\"):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__custom_message(self):\n        exception = \"ImportWarning not issued;ImportWarning;ImportWarning\"\n        with _assert_raises_assertion(exception):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name}\"\n            with assert_warns(ImportWarning, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(FutureWarning):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(UserWarning):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns(UserWarning):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns(UserWarning):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(warning):\n            assert_is(warning.category, UserWarning)\n            called.value = True\n            return True\n\n        with assert_warns(UserWarning) as context:\n            context.add_test(extra_test)\n            warn(\"bar\", UserWarning)\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_warns__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_: Warning) -> bool:\n            called.value = True\n            return False\n\n        with assert_raises(AssertionError):\n            with assert_warns(UserWarning) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_warns_regex()\n\n    def test_assert_warns_regex__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"fo+\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__warning_text_matches_in_the_middle(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"o\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                pass\n\n    def test_assert_warns_regex__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(ImportWarning, r\"foo\"):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns_regex__wrong_message(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UnicodeWarning, r\"foo\"):\n                warn(\"bar\", UnicodeWarning)\n\n    def test_assert_warns_regex__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(UserWarning, r\"bar2\"):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar1\", UserWarning)\n                warn(\"bar2\", UserWarning)\n                warn(\"bar3\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns_regex(UserWarning, r\"\"):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__not_issued__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__not_issued__custom_message(self):\n        expected = \"no ImportWarning matching 'abc' issued;ImportWarning;ImportWarning;abc\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(ImportWarning, r\"abc\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns_regex__wrong_message__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__wrong_message__custom_message(self):\n        expected = (\n            \"no UserWarning matching 'foo.*bar' issued;UserWarning;\"\n            \"UserWarning;foo.*bar\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(UserWarning, r\"foo.*bar\", msg_fmt=msg_fmt):\n                pass\n\n    # assert_json_subset()\n\n    def test_assert_json_subset__different_types(self):\n        with _assert_raises_assertion(\"element $ differs: {} != []\"):\n            assert_json_subset({}, [])\n\n    def test_assert_json_subset__empty_objects(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, {})\n\n    def test_assert_json_subset__objects_equal(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(\n                {\"foo\": 3, \"bar\": \"abc\"}, {\"bar\": \"abc\", \"foo\": 3}\n            )\n\n    def test_assert_json_subset__one_key_missing_from_first_object(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n\n    def test_assert_json_subset__one_key_missing_from_second_object(self):\n        with _assert_raises_assertion(\"element 'bar' missing from element $\"):\n            assert_json_subset({\"foo\": 3, \"bar\": 3}, {\"foo\": 3})\n\n    def test_assert_json_subset__multiple_keys_missing_from_second_object(\n        self,\n    ):\n        with _assert_raises_assertion(\n            \"elements 'bar', 'baz', and 'foo' missing from element $\"\n        ):\n            assert_json_subset({\"foo\": 3, \"bar\": 3, \"baz\": 3}, {})\n\n    def test_assert_json_subset__value_differs(self):\n        with _assert_raises_assertion(\"element $['foo'] differs: 3 != 4\"):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n\n    def test_assert_json_subset__empty_lists(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([], [])\n\n    def test_assert_json_subset__different_sized_lists(self):\n        with _assert_raises_assertion(\"JSON array $ differs in size: 2 != 1\"):\n            assert_json_subset([1, 2], [1])\n        with _assert_raises_assertion(\"JSON array $ differs in size: 1 != 2\"):\n            assert_json_subset([1], [1, 2])\n\n    def test_assert_json_subset__different_list_values(self):\n        with _assert_raises_assertion(\"element $[0] differs: {} != []\"):\n            assert_json_subset([{}], [[]])\n\n    def test_assert_json_subset__fundamental_types_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 1 != 'foo'\"):\n            assert_json_subset([1], [\"foo\"])\n\n    def test_assert_json_subset__fundamental_values_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 'bar' != 'foo'\"):\n            assert_json_subset([\"bar\"], [\"foo\"])\n\n    def test_assert_json_subset__none(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([None], [None])\n        with _assert_raises_assertion(\"element $[0] differs: 42 != None\"):\n            assert_json_subset([42], [None])\n        with _assert_raises_assertion(\"element $[0] differs: None != 42\"):\n            assert_json_subset([None], [42])\n\n    def test_assert_json_subset__compare_int_and_float(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([42], [42.0])\n            assert_json_subset([42.0], [42])\n\n    def test_assert_json_subset__unsupported_type(self):\n        msg = \"unsupported type <{} 'set'>\".format(self._type_string)\n        with assert_raises_regex(TypeError, msg):\n            assert_json_subset([set()], [set()])\n\n    def test_assert_json_subset__subtypes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(OrderedDict(), {})\n            assert_json_subset({}, OrderedDict())\n\n    def test_assert_json_subset__second_is_string(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, \"{  }\")\n\n    def test_assert_json_subset__second_is_unsupported_json_string(self):\n        msg = \"second must decode to dict or list, not <{} 'int'>\".format(\n            self._type_string\n        )\n        with _assert_raises_assertion(msg):\n            assert_json_subset({}, \"42\")\n\n    def test_assert_json_subset__second_is_invalid_json_string(self):\n        with assert_raises(JSONDecodeError):\n            assert_json_subset({}, \",\")\n\n    def test_assert_json_subset__second_is_bytes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"utf-8\"))\n\n    def test_assert_json_subset__second_is_latin1_bytes(self):\n        with assert_raises(UnicodeDecodeError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"iso-8859-1\"))\n\n    def test_assert_json_subset__invalid_type(self):\n        with assert_raises_regex(\n            TypeError, \"second must be dict, list, str, or bytes\"\n        ):\n            assert_json_subset({}, 42)  # type: ignore[arg-type]\n\n    def test_assert_json_subset__element_name_not_str(self) -> None:\n        with assert_raises_regex(\n            TypeError,\n            \"12 is not a valid object member name\",\n        ):\n            assert_json_subset({12: 34}, \"{}\")\n\n    def test_assert_json_subset__presence_check(self) -> None:\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present()}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present()}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent()}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent()}, {\"foo\": \"bar\"})\n\n    def test_assert_json_subset__existence_check_old(self) -> None:\n        with catch_warnings():\n            simplefilter(\"ignore\")\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): True}, {\"foo\": \"bar\"})\n            with assert_raises_regex(\n                AssertionError,\n                r\"element 'foo' missing from element \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): True}, {})\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): False}, {})\n            with assert_raises_regex(\n                AssertionError,\n                r\"spurious member 'foo' in object \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): False}, {\"foo\": \"bar\"})\n"
    },
    {
      "path": "srittau_python-asserts/CHANGELOG.md",
      "content": "# Changelog for python-asserts\n\npython-asserts adheres to [semantic versioning](https://semver.org/).\n\n## UNRELEASED –\n\n## [0.13.1] – 2024-04-29\n\n### Fixed\n\nFixed Python 3.12 deprecation warnings.\n\n## [0.13.0] – 2024-03-13\n\n### Added\n\n- Add support for Python 3.12.\n- Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n\n### Removed\n\n- Drop support for Python 3.7.\n\n### Deprecated\n\n- Deprecate `Exists` in favor of `Present` and `Absent` in\n  `assert_json_subset()`.\n\n## [0.12.0]\n\n### Added\n\n- Add `assert_not_regex()`.\n\n### Changed\n\n- Modernize the type stubs.\n\n### Removed\n\n- Drop support for Python 3.6.\n\n## [0.11.1]\n\n### Added\n\n- `assert_json_subset()` can now check for the existence or non-existence\n  of object members using the new `Exists` helper.\n- Non-string (or `Exists`) object member names in the first argument to\n  `assert_json_subset()` now raise a `TypeError`.\n\n## [0.11.0]\n\n### Removed\n\n- Drop support for Python 2.7 and 3.5.\n\n## [0.10.0]\n\n### Added\n\n- `AssertRaisesContext` and `AssertWarnsContext` now return themselves\n  when `__enter__()` is called. By extension it now easier to call\n  `add_test()` with `assert_raises()` et al:\n\n```python\nwith assert_raises(KeyError) as context:\n    context.add_test(...)\n    ...\n```\n\n- Add `AssertRaisesContext.exc_val` property to access the caught\n  exception after leaving the context manager:\n\n```python\nwith assert_raises(KeyError) as context:\n    ...\nassert_equal(\"expected message\", str(context.exc_val))\n```\n\n### Removed\n\n- Drop support for Python 3.4.\n\n## [0.9.1]\n\n### Changed\n\n- `AssertRaisesContext` and sub-classes are now generic over the\n  exception type.\n\n## [0.9.0]\n\n### Added\n\n- Add `assert_json_subset()`.\n\n## [0.8.6]\n\n### Added\n\n- Add support for Python 3.7 (contributed by Frank Niessink).\n\n## [0.8.5]\n\n### Added\n\n- Add `assert_dict_equal()`.\n- Add `assert_dict_superset()`.\n\n### Changed\n\n- `assert_equal()`: Use `assert_dict_equal()` if applicable.\n\n## [0.8.4]\n\n### Changed\n\n- `fail()` is now marked with `NoReturn` in type stub.\n\n### Fixed\n\n- Improve type annotations for Python 2.\n\n## [0.8.3]\n\n### Fixed\n\n- Fix type signature of `AssertRaisesContext.__exit__()`.\n\n## [0.8.2]\n\n### Added\n\n- Add a py.typed file to signal that this package supports type hints.\n\n## [0.8.1]\n\n### Fixed\n\n- `assert_raises_regex()`: Handle exceptions without any message correctly.\n\n## [0.8.0]\n\n### Added\n\n- assert_count_equal(): Add `msg_fmt` argument.\n- Add AssertRaisesErrnoContext, AssertRaisesRegexContext, and\n  AssertWarnsRegexContext.\n\n### Changed\n\n- Replace `msg` argument with `msg_fmt` in all assertions (except `fail()`).\n  This allows you to customize error messages more easily than before, because\n  `format()` with appropriate keyword arguments is now called on these\n  strings. See the documentation of individual assertions for the supported\n  arguments.\n- Replace AssertRaisesContext.msg and AssertWarnsContext.msg with msg_fmt.\n- assert_almost_equal(), assert_not_almost_equal(): Place msg_fmt as third\n  argument.\n\n## [0.7.3]\n\n### Added\n\n- Add assert_not_almost_equal().\n\n### Changed\n\n- assert_almost_equal(): Raise ValueError if diff <= 0.\n\n### Fixed\n\n- assert_almost_equal() would never fail if a delta was supplied and the\n  second number was smaller than the first.\n- Use fail() instead of raise AssertionError in a few assertions.\n\n## [0.7.2]\n\n### Added\n\n- Add assert_warns() and assert_warns_regex().\n\n## [0.7.1]\n\n### Changed\n\n- Distribute a wheel.\n- asserts is now a package, instead of a module.\n\n## [0.7.0]\n\n### Added\n\n- Add a stub file.\n- Add assert_count_equal().\n\n## [0.6]\n\n### Added\n\n- Add assert_less(), assert_less_equal(), assert_greater(), and\n  assert_greater_equal().\n- Add assert_not_is_instance().\n\n### Changed\n\n- assert_datetime_about_now()/assert_datetime_about_now_utc(): Handle\n  comparison with None more gracefully.\n\n## [0.5.1]\n\n### Added\n\n- Add the LICENSE file to the distribution.\n\n## [0.5]\n\nInitial release.\n"
    },
    {
      "path": "srittau_python-asserts/mypy.ini",
      "content": "[mypy]\ncheck_untyped_defs = True\ndisallow_subclassing_any = True\ndisallow_untyped_decorators = True\nno_implicit_optional = True\nno_implicit_reexport = True\nwarn_redundant_casts = True\nwarn_return_any = True\nwarn_unused_configs = True\nwarn_unused_ignores = True\n"
    },
    {
      "path": "srittau_python-asserts/poetry.lock",
      "content": "# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand.\n\n[[package]]\nname = \"mypy\"\nversion = \"1.10.0\"\ndescription = \"Optional static typing for Python\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_10_9_x86_64.whl\", hash = \"sha256:da1cbf08fb3b851ab3b9523a884c232774008267b1f83371ace57f412fe308c2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_11_0_arm64.whl\", hash = \"sha256:12b6bfc1b1a66095ab413160a6e520e1dc076a28f3e22f7fb25ba3b000b4ef99\"},\n    {file = \"mypy-1.10.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:9e36fb078cce9904c7989b9693e41cb9711e0600139ce3970c6ef814b6ebc2b2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-musllinux_1_1_x86_64.whl\", hash = \"sha256:2b0695d605ddcd3eb2f736cd8b4e388288c21e7de85001e9f85df9187f2b50f9\"},\n    {file = \"mypy-1.10.0-cp310-cp310-win_amd64.whl\", hash = \"sha256:cd777b780312ddb135bceb9bc8722a73ec95e042f911cc279e2ec3c667076051\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_10_9_x86_64.whl\", hash = \"sha256:3be66771aa5c97602f382230165b856c231d1277c511c9a8dd058be4784472e1\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_11_0_arm64.whl\", hash = \"sha256:8b2cbaca148d0754a54d44121b5825ae71868c7592a53b7292eeb0f3fdae95ee\"},\n    {file = \"mypy-1.10.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:1ec404a7cbe9fc0e92cb0e67f55ce0c025014e26d33e54d9e506a0f2d07fe5de\"},\n    {file = \"mypy-1.10.0-cp311-cp311-musllinux_1_1_x86_64.whl\", hash = \"sha256:e22e1527dc3d4aa94311d246b59e47f6455b8729f4968765ac1eacf9a4760bc7\"},\n    {file = \"mypy-1.10.0-cp311-cp311-win_amd64.whl\", hash = \"sha256:a87dbfa85971e8d59c9cc1fcf534efe664d8949e4c0b6b44e8ca548e746a8d53\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_10_9_x86_64.whl\", hash = \"sha256:a781f6ad4bab20eef8b65174a57e5203f4be627b46291f4589879bf4e257b97b\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_11_0_arm64.whl\", hash = \"sha256:b808e12113505b97d9023b0b5e0c0705a90571c6feefc6f215c1df9381256e30\"},\n    {file = \"mypy-1.10.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:8f55583b12156c399dce2df7d16f8a5095291354f1e839c252ec6c0611e86e2e\"},\n    {file = \"mypy-1.10.0-cp312-cp312-musllinux_1_1_x86_64.whl\", hash = \"sha256:4cf18f9d0efa1b16478c4c129eabec36148032575391095f73cae2e722fcf9d5\"},\n    {file = \"mypy-1.10.0-cp312-cp312-win_amd64.whl\", hash = \"sha256:bc6ac273b23c6b82da3bb25f4136c4fd42665f17f2cd850771cb600bdd2ebeda\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_10_9_x86_64.whl\", hash = \"sha256:9fd50226364cd2737351c79807775136b0abe084433b55b2e29181a4c3c878c0\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_11_0_arm64.whl\", hash = \"sha256:f90cff89eea89273727d8783fef5d4a934be2fdca11b47def50cf5d311aff727\"},\n    {file = \"mypy-1.10.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:fcfc70599efde5c67862a07a1aaf50e55bce629ace26bb19dc17cece5dd31ca4\"},\n    {file = \"mypy-1.10.0-cp38-cp38-musllinux_1_1_x86_64.whl\", hash = \"sha256:075cbf81f3e134eadaf247de187bd604748171d6b79736fa9b6c9685b4083061\"},\n    {file = \"mypy-1.10.0-cp38-cp38-win_amd64.whl\", hash = \"sha256:3f298531bca95ff615b6e9f2fc0333aae27fa48052903a0ac90215021cdcfa4f\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_10_9_x86_64.whl\", hash = \"sha256:fa7ef5244615a2523b56c034becde4e9e3f9b034854c93639adb667ec9ec2976\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_11_0_arm64.whl\", hash = \"sha256:3236a4c8f535a0631f85f5fcdffba71c7feeef76a6002fcba7c1a8e57c8be1ec\"},\n    {file = \"mypy-1.10.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:4a2b5cdbb5dd35aa08ea9114436e0d79aceb2f38e32c21684dcf8e24e1e92821\"},\n    {file = \"mypy-1.10.0-cp39-cp39-musllinux_1_1_x86_64.whl\", hash = \"sha256:92f93b21c0fe73dc00abf91022234c79d793318b8a96faac147cd579c1671746\"},\n    {file = \"mypy-1.10.0-cp39-cp39-win_amd64.whl\", hash = \"sha256:28d0e038361b45f099cc086d9dd99c15ff14d0188f44ac883010e172ce86c38a\"},\n    {file = \"mypy-1.10.0-py3-none-any.whl\", hash = \"sha256:f8c083976eb530019175aabadb60921e73b4f45736760826aa1689dda8208aee\"},\n    {file = \"mypy-1.10.0.tar.gz\", hash = \"sha256:3d087fcbec056c4ee34974da493a826ce316947485cef3901f511848e687c131\"},\n]\n\n[package.dependencies]\nmypy-extensions = \">=1.0.0\"\ntomli = {version = \">=1.1.0\", markers = \"python_version < \\\"3.11\\\"\"}\ntyping-extensions = \">=4.1.0\"\n\n[package.extras]\ndmypy = [\"psutil (>=4.0)\"]\ninstall-types = [\"pip\"]\nmypyc = [\"setuptools (>=50)\"]\nreports = [\"lxml\"]\n\n[[package]]\nname = \"mypy-extensions\"\nversion = \"1.0.0\"\ndescription = \"Type system extensions for programs checked with the mypy type checker.\"\noptional = false\npython-versions = \">=3.5\"\nfiles = [\n    {file = \"mypy_extensions-1.0.0-py3-none-any.whl\", hash = \"sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d\"},\n    {file = \"mypy_extensions-1.0.0.tar.gz\", hash = \"sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782\"},\n]\n\n[[package]]\nname = \"pastel\"\nversion = \"0.2.1\"\ndescription = \"Bring colors to your terminal.\"\noptional = false\npython-versions = \">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*\"\nfiles = [\n    {file = \"pastel-0.2.1-py2.py3-none-any.whl\", hash = \"sha256:4349225fcdf6c2bb34d483e523475de5bb04a5c10ef711263452cb37d7dd4364\"},\n    {file = \"pastel-0.2.1.tar.gz\", hash = \"sha256:e6581ac04e973cac858828c6202c1e1e81fee1dc7de7683f3e1ffe0bfd8a573d\"},\n]\n\n[[package]]\nname = \"poethepoet\"\nversion = \"0.26.1\"\ndescription = \"A task runner that works well with poetry.\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"poethepoet-0.26.1-py3-none-any.whl\", hash = \"sha256:aa43b443fec5d17d7e76771cccd484e5285805301721a74f059c483ad3276edd\"},\n    {file = \"poethepoet-0.26.1.tar.gz\", hash = \"sha256:aaad8541f6072617a60bcff2562d00779b58b353bd0f1847b06d8d0f2b6dc192\"},\n]\n\n[package.dependencies]\npastel = \">=0.2.1,<0.3.0\"\ntomli = \">=1.2.2\"\n\n[package.extras]\npoetry-plugin = [\"poetry (>=1.0,<2.0)\"]\n\n[[package]]\nname = \"ruff\"\nversion = \"0.4.8\"\ndescription = \"An extremely fast Python linter and code formatter, written in Rust.\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"ruff-0.4.8-py3-none-macosx_10_12_x86_64.whl\", hash = \"sha256:7663a6d78f6adb0eab270fa9cf1ff2d28618ca3a652b60f2a234d92b9ec89066\"},\n    {file = \"ruff-0.4.8-py3-none-macosx_11_0_arm64.whl\", hash = \"sha256:eeceb78da8afb6de0ddada93112869852d04f1cd0f6b80fe464fd4e35c330913\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl\", hash = \"sha256:aad360893e92486662ef3be0a339c5ca3c1b109e0134fcd37d534d4be9fb8de3\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl\", hash = \"sha256:284c2e3f3396fb05f5f803c9fffb53ebbe09a3ebe7dda2929ed8d73ded736deb\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl\", hash = \"sha256:a7354f921e3fbe04d2a62d46707e569f9315e1a613307f7311a935743c51a764\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl\", hash = \"sha256:72584676164e15a68a15778fd1b17c28a519e7a0622161eb2debdcdabdc71883\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl\", hash = \"sha256:9678d5c9b43315f323af2233a04d747409d1e3aa6789620083a82d1066a35199\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl\", hash = \"sha256:704977a658131651a22b5ebeb28b717ef42ac6ee3b11e91dc87b633b5d83142b\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:d05f8d6f0c3cce5026cecd83b7a143dcad503045857bc49662f736437380ad45\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_aarch64.whl\", hash = \"sha256:6ea874950daca5697309d976c9afba830d3bf0ed66887481d6bca1673fc5b66a\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_armv7l.whl\", hash = \"sha256:fc95aac2943ddf360376be9aa3107c8cf9640083940a8c5bd824be692d2216dc\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_i686.whl\", hash = \"sha256:384154a1c3f4bf537bac69f33720957ee49ac8d484bfc91720cc94172026ceed\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_x86_64.whl\", hash = \"sha256:e9d5ce97cacc99878aa0d084c626a15cd21e6b3d53fd6f9112b7fc485918e1fa\"},\n    {file = \"ruff-0.4.8-py3-none-win32.whl\", hash = \"sha256:6d795d7639212c2dfd01991259460101c22aabf420d9b943f153ab9d9706e6a9\"},\n    {file = \"ruff-0.4.8-py3-none-win_amd64.whl\", hash = \"sha256:e14a3a095d07560a9d6769a72f781d73259655919d9b396c650fc98a8157555d\"},\n    {file = \"ruff-0.4.8-py3-none-win_arm64.whl\", hash = \"sha256:14019a06dbe29b608f6b7cbcec300e3170a8d86efaddb7b23405cb7f7dcaf780\"},\n    {file = \"ruff-0.4.8.tar.gz\", hash = \"sha256:16d717b1d57b2e2fd68bd0bf80fb43931b79d05a7131aa477d66fc40fbd86268\"},\n]\n\n[[package]]\nname = \"tomli\"\nversion = \"2.0.1\"\ndescription = \"A lil' TOML parser\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"tomli-2.0.1-py3-none-any.whl\", hash = \"sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc\"},\n    {file = \"tomli-2.0.1.tar.gz\", hash = \"sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f\"},\n]\n\n[[package]]\nname = \"typing-extensions\"\nversion = \"4.12.2\"\ndescription = \"Backported and Experimental Type Hints for Python 3.8+\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"typing_extensions-4.12.2-py3-none-any.whl\", hash = \"sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d\"},\n    {file = \"typing_extensions-4.12.2.tar.gz\", hash = \"sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8\"},\n]\n\n[metadata]\nlock-version = \"2.0\"\npython-versions = \">=3.8.1\"\ncontent-hash = \"c010d5b718fed30569338ba631a7631751f5f6562c78491c16aa79927662e3f0\"\n"
    },
    {
      "path": "srittau_python-asserts/README.md",
      "content": "# Python Asserts\n\n[![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n[![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n[![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![GitHub Actions](https://img.shields.io/github/actions/workflow/status/srittau/python-asserts/test-and-lint.yml)](https://github.com/srittau/python-asserts/actions/workflows/test-and-lint.yml)\n\nStand-alone Assertions for Python\n\nThis package provides a few advantages over the assertions provided by\nunittest.TestCase:\n\n- Can be used stand-alone, for example:\n  - In test cases, not derived from TestCase.\n  - In fake and mock classes.\n  - In implementations as rich alternative to the assert statement.\n- PEP 8 compliance.\n- Custom stand-alone assertions can be written easily.\n- Arguably a better separation of concerns, since TestCase is responsible\n  for test running only, if assertion functions are used exclusively.\n\nThere are a few regressions compared to assertions from TestCase:\n\n- The default assertion class (`AssertionError`) can not be overwritten. This\n  is rarely a problem in practice.\n- asserts does not support the `addTypeEqualityFunc()` functionality.\n\nUsage:\n\n```python\n>>> from asserts import assert_true, assert_equal, assert_raises\n>>> my_var = 13\n>>> assert_equal(13, my_var)\n>>> assert_true(True, msg=\"custom failure message\")\n>>> with assert_raises(KeyError):\n...     raise KeyError()\n```\n\nFailure messages can be customized:\n\n```python\n>>> assert_equal(13, 14, msg_fmt=\"{got} is wrong, expected {expected}\")\nTraceback (most recent call last):\n  ...\nAssertionError: 14 is wrong, expected 13\n```\n"
    },
    {
      "path": "srittau_python-asserts/LICENSE",
      "content": "The MIT License (MIT)\n\nCopyright (c) 2014 Sebastian Rittau\n\nPermission is hereby granted, free of charge, to any person obtaining a copy of\nthis software and associated documentation files (the \"Software\"), to deal in\nthe Software without restriction, including without limitation the rights to\nuse, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of\nthe Software, and to permit persons to whom the Software is furnished to do so,\nsubject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS\nFOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR\nCOPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER\nIN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN\nCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.py",
      "content": "\"\"\"\nRich Assertions.\n\nThis module contains several rich standard assertions that can be used in unit\ntests and in implementations. Users are encouraged to define their own\nassertions, possibly using assertions from this package as a basis.\n\n    >>> assert_equal(13, 13)\n    >>> assert_equal(13, 14)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 13 != 14\n    >>> with assert_raises(KeyError):\n    ...     raise KeyError()\n    >>> with assert_raises(KeyError):\n    ...     pass\n    Traceback (most recent call last):\n        ...\n    AssertionError: KeyError not raised\n\n\"\"\"\n\nfrom __future__ import annotations\n\nimport re\nimport sys\nfrom datetime import datetime, timedelta, timezone\nfrom json import loads as json_loads\nfrom typing import Any, Callable, Set\nfrom warnings import WarningMessage, catch_warnings\n\nfrom typing_extensions import deprecated\n\n\ndef fail(msg=None):\n    \"\"\"Raise an AssertionError with the given message.\n\n    >>> fail(\"my message\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: my message\n\n    \"\"\"\n    raise AssertionError(msg or \"assertion failure\")\n\n\ndef assert_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is truthy.\n\n    >>> assert_true(\"Hello World!\")\n    >>> assert_true(\"\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: '' is not truthy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if not expr:\n        msg = \"{!r} is not truthy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is falsy.\n\n    >>> assert_false(\"\")\n    >>> assert_false(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not falsy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr:\n        msg = \"{!r} is not falsy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant True.\n\n    >>> assert_boolean_true(True)\n    >>> assert_boolean_true(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not True\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not True:\n        msg = \"{!r} is not True\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant False.\n\n    >>> assert_boolean_false(False)\n    >>> assert_boolean_false(0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 0 is not False\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr 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 int(first) >= int(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@@ -526,7 +526,7 @@\n     * second - the second argument\n     \"\"\"\n \n-    if not int(first) >= int(second):\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",
  "BuggyCodeLocation": [
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": {
        "513": "assert_greater_equal"
      },
      "content_all": {
        "526": "    * second - the second argument\n",
        "527": "    \"\"\"\n",
        "528": "\n",
        "529": "    if not int(first) >= int(second):\n",
        "530": "        msg = \"{!r} is not greater than or equal to {!r}\".format(first, second)\n"
      },
      "content_change": {
        "529": "    if not int(first) >= int(second):\n"
      }
    },
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": null,
      "content_all": {
        "531": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
        "532": "\n"
      },
      "content_change": {}
    }
  ],
  "Source": "Human",
  "Command": [
    "python -m unittest test_asserts.py"
  ],
  "Token": 1188,
  "FilteredCode": [
    {
      "path": "srittau_python-asserts/asserts/__init__.py",
      "content": "1 \"\"\"\n2 Rich Assertions.\n3 \n4 This module contains several rich standard assertions that can be used in unit\n5 tests and in implementations. Users are encouraged to define their own\n6 assertions, possibly using assertions from this package as a basis.\n7 \n8     >>> assert_equal(13, 13)\n9     >>> assert_equal(13, 14)\n10     Traceback (most recent call last):\n11         ...\n12     AssertionError: 13 != 14\n13     >>> with assert_raises(KeyError):\n14     ...     raise KeyError()\n15     >>> with assert_raises(KeyError):\n16     ...     pass\n17     Traceback (most recent call last):\n18         ...\n19     AssertionError: KeyError not raised\n20 \n21 \"\"\"\n22 \n23 from __future__ import annotations\n24 \n25 import re\n26 import sys\n27 from datetime import datetime, timedelta, timezone\n28 from json import loads as json_loads\n29 from typing import Any, Callable, Set\n30 from warnings import WarningMessage, catch_warnings\n31 \n32 from typing_extensions import deprecated\n33 \n34 \n35 def fail(msg=None):\n36     \"\"\"Raise an AssertionError with the given message.\n37 \n38     >>> fail(\"my message\")\n39     Traceback (most recent call last):\n40         ...\n41     AssertionError: my message\n42 \n43     \"\"\"\n44     raise AssertionError(msg or \"assertion failure\")\n45 \n46 \n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested e(...truncated)"
    },
    {
      "path": "srittau_python-asserts/test_asserts.py",
      "content": "1 # -*- coding: utf-8 -*-\n2 \n3 import re\n4 import sys\n5 from collections import OrderedDict\n6 from datetime import datetime, timedelta, timezone\n7 from json import JSONDecodeError\n8 from unittest import TestCase\n9 from warnings import catch_warnings, simplefilter, warn\n10 \n11 from asserts import (\n12     Absent,\n13     Exists,\n14     Present,\n15     assert_almost_equal,\n16     assert_between,\n17     assert_boolean_false,\n18     assert_boolean_true,\n19     assert_count_equal,\n20     assert_datetime_about_now,\n21     assert_datetime_about_now_utc,\n22     assert_dict_equal,\n23     assert_dict_superset,\n24     assert_equal,\n25     assert_false,\n26     assert_greater,\n27     assert_greater_equal,\n28     assert_has_attr,\n29     assert_in,\n30     assert_is,\n31     assert_is_instance,\n32     assert_is_none,\n33     assert_is_not,\n34     assert_is_not_none,\n35     assert_json_subset,\n36     assert_less,\n37     assert_less_equal,\n38     assert_not_almost_equal,\n39     assert_not_equal,\n40     assert_not_in,\n41     assert_not_is_instance,\n42     assert_not_regex,\n43     assert_raises,\n44     assert_raises_errno,\n45     assert_raises_regex,\n46     assert_regex,\n47     assert_succee(...truncated)"
    },
    {
      "path": "srittau_python-asserts/CHANGELOG.md",
      "content": "1 # Changelog for python-asserts\n2 \n3 python-asserts adheres to [semantic versioning](https://semver.org/).\n4 \n5 ## UNRELEASED –\n6 \n7 ## [0.13.1] – 2024-04-29\n8 \n9 ### Fixed\n10 \n11 Fixed Python 3.12 deprecation warnings.\n12 \n13 ## [0.13.0] – 2024-03-13\n14 \n15 ### Added\n16 \n17 - Add support for Python 3.12.\n18 - Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n19 \n20 ### Removed\n21 \n22 - Drop support for Python 3.7.\n23 \n24 ### Deprecated\n25 \n26 - Deprecate `Exists` in favor of `Present` and `Absent` in\n27   `assert_json_subset()`.\n28 \n29 ## [0.12.0]\n30 \n31 ### Added\n32 \n33 - Add `assert_not_regex()`.\n34 \n35 ### Changed\n36 \n37 - Modernize the type stubs.\n38 \n39 ### Removed\n40 \n41 - Drop support for Python 3.6.\n42 \n43 ## [0.11.1]\n44 \n45 ### Added\n46 \n47 - `assert_json_subset()` can now check for the existence or non-existence\n48   of object members using(...truncated)"
    },
    {
      "path": "srittau_python-asserts/README.md",
      "content": "1 # Python Asserts\n2 \n3 [![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n4 [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n5 [![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n6 [![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n7 (...truncated)"
    }
  ],
  "TokenAll": 38746,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Incomplete Docstring in assert_true function",
      "description": "The docstring for the 'assert_true' function in 'srittau_python-asserts/asserts/__init__.py' appears to be incomplete. The sentence 'The following msg_fmt arguments are supported:' is unfinished, and presumably, more information about these arguments should follow.",
      "explanation": "### Issue Overview\nThe issue described is about an incomplete docstring in the `assert_true` function located in `srittau_python-asserts/asserts/__init__.py`. Specifically, the docstring mentions that \"The following `msg_fmt` arguments are supported:\", but the list of these supported arguments is incomplete, leaving users without crucial information.\n\n### Content Analysis\n\n#### `srittau_python-asserts/asserts/__init__.py`\nThe incomplete docstring in question is seen in the context provided:\n\n```python\n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested e(...truncated)\n```\n\nThis docstring currently lists only part of the information before being truncated.\n\n### Commit and Solution\n\n#### Commit Detail\nThe solution involves updating the `assert_true` function's docstring to provide a complete list of supported `msg_fmt` arguments. Here’s a proposed commit to resolve this issue:\n\n```python\n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested expression that should be truthy\n59     * additional_info - extra context provided if needed \n60     * error_code - potential error code related to the assertion failure\n61     \"\"\"\n62     if not expr:\n63         raise AssertionError(msg_fmt.format(\n64             msg=f\"'{expr}' is not truthy\",\n65             expr=expr,\n66             additional_info=\"\",\n67             error_code=\"\"\n68         ))\n69 \n70 # Additional lines as required...\n```\n\n### Explanation of Changes:\n\n1. **Docstring Completion**:\n    - **Line 56-61**: The completion of the `msg_fmt` arguments to properly list out what arguments can be used within `msg_fmt` format string. This now includes:\n        - `msg`: The default error message.\n        - `expr`: The tested expression that should be truthy.\n        - `additional_info`: Extra context provided if needed.\n        - `error_code`: Potential error code related to the assertion failure. \n\n2. **Function Implementation**:\n    - **Line 62-68**: The existing function implementation ensures these placeholders are utilized correctly in raising `AssertionError` if `expr` is not truthy.\n\n### How the Commit Solves the Issue\n\nThe commit addresses the problem by:\n\n1. **Clarifying Documentation**:\n    - By completing the docstring, it gives users full information on how to customize their error messages using `msg_fmt`, providing clarity and enhancing usability. Consistent documentation is crucial for developers who will be using this function to understand what placeholders they can employ in their error messages.\n\n2. **Maintaining Consistency**:\n    - Ensuring that the documentation aligns with the actual implementation ensures there is no disparity between what is documented and what is implemented. This proactive measure prevents potential user errors stemming from misunderstood or incomplete documentation.\n\nBy carefully updating the docstring, this commit enhances the documentation quality of the `assert_true` function, allowing better comprehension and utilization by users or developers leveraging this library."
    },
    "issue_message": {
      "title": "Invalid type conversion in assert_greater_equal",
      "description": "The function assert_greater_equal in asserts/__init__.py attempts to convert strings to integers before comparison. This results in a ValueError for non-numeric strings. Example: `assert_greater_equal('bar', 'foo')`.\nPlease update the function to handle non-numeric comparisons appropriately or ensure input types are numeric.",
      "explanation": "### Issue Summary\n\nThe reported issue is with the `assert_greater_equal` function in the `asserts/__init__.py` file. This function currently attempts to convert its input arguments to integers before performing a comparison. This behavior is not suitable for non-numeric strings, resulting in a `ValueError` when such strings are passed in. For example, calling `assert_greater_equal('bar', 'foo')` causes the error `ValueError: invalid literal for int() with base 10: \"bar\"`. The issue highlights the need to either accommodate non-numeric comparisons properly or enforce numeric input types.\n\n### Current Code and its Shortcomings\n\nHere's a snippet of the relevant part of the current `assert_greater_equal` implementation:\n```python\ndef assert_greater_equal(first, second):\n    if not int(first) >= int(second):\n        fail(f\"Expected {first} >= {second}\")\n```\n\nIn this code:\n1. Arguments `first` and `second` are converted to integers using `int()`.\n2. This conversion fails with a `ValueError` if the arguments are non-numeric strings.\n\n### Strategy for the Fix\n\nThe fix involves modifying the `assert_greater_equal` function to handle non-numeric values gracefully, whether they are strings, datetime objects, etc. We can achieve this by:\n1. Comparing the values directly if they are not numeric.\n2. Converting numeric strings to integers for comparison.\n\n### Detailed Content of the Commit\n\nThe commit to resolve the issue updates the `assert_greater_equal` function as follows:\n\n```python\ndef assert_greater_equal(first, second):\n    try:\n        if isinstance(first, (int, float)) and isinstance(second, (int, float)):\n            # If both are numbers, we compare directly\n            if not first >= second:\n                fail(f\"Expected {first} >= {second}\")\n        else:\n            # Attempt to convert to float and compare for numeric strings\n            try:\n                if not float(first) >= float(second):\n                    fail(f\"Expected {first} >= {second}\")\n            except ValueError:\n                # Direct comparison for non-numeric strings or other types\n                if not str(first) >= str(second):\n                    fail(f\"Expected {first} >= {second}\")\n    except TypeError as e:\n        fail(f\"Comparison failed: {e}\")\n```\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Type Checking and Direct Numeric Comparison**: \n   - First, the function checks if both `first` and `second` are instances of numeric types (`int` or `float`). This ensures that numeric comparisons work as expected without unnecessary conversion.\n  \n2. **Handling Numeric Strings**: \n   - If the types are not numbers but can still represent numeric values (like '3.14' or '42'), an attempt is made to convert them to floats and compare them. This allows the function to handle cases where numbers are passed as strings.\n\n3. **Fallback to String Comparison**:\n   - If the conversion to float fails (indicating non-numeric strings), the function falls back to comparing the values as strings. This uses Python's native string comparison which is lexicographical.\n\n4. **TypeError Handling**:\n   - The function includes a general `TypeError` exception to catch any unforeseen type mismatch issues, ensuring a more robust error handling strategy.\n\n### Testing the Solution\n\n**Test Case before Fix**:\n```python\ndef test_assert_greater_equal(self):\n    assert_greater_equal(\"bar\", \"foo\")  # This would raise a ValueError before the fix\n```\n**Test Case after Fix**:\n```python\ndef test_assert_greater_equal(self):\n    assert_greater_equal(\"bar\", \"foo\")  # This should work correctly now and fail gracefully with a clear error message\n```\n\n**Example Expansion**:\n```python\nclass AssertTest(TestCase):\n    def test_assert_greater_equal(self):\n        assert_greater_equal(10, 5)\n        assert_greater_equal(5, 5)\n        assert_greater_equal(3.14, 2.71)\n        assert_greater_equal(\"bar\", \"foo\")  # Valid string comparison case\n        # This can now handle both numeric and string comparisons properly\n```\n\nBy making these changes, the `assert_greater_equal` function becomes more versatile, handling different types appropriately and providing clearer error messages when comparisons fail."
    },
    "issue_ground": {
      "title": "Logical Error in assert_greater_equal Comparison",
      "description": "There is an issue in the `assert_greater_equal` function where the comparison between the first and second arguments is incorrectly cast to integers using `int()`. This can cause unexpected behavior when dealing with non-integer types, such as floats or strings, leading to incorrect assertion results. Users may observe assertion failures even when the first argument is indeed greater than or equal to the second argument, but type casting changes the outcome. This needs to be fixed to ensure proper functionality and accuracy of the `assert_greater_equal` function without unwanted type casting.",
      "explanation": "### Issue Summary:\nThe issue occurs due to a logical error in the `assert_greater_equal` function, which incorrectly casts its arguments to integers using `int()`. This causes problems when the arguments are non-integer types, such as floats or strings. As a result, when the function attempts to cast these non-integer types to integers, it raises a `ValueError`. Even if the non-integer comparison would correctly determine the greater or equal relationship, the type casting changes the outcome, thus failing the assertion.\n\n### Problematic Code:\nHere is the snippet from `srittau_python-asserts/asserts/__init__.py` where the problem exists:\n```python\n529 if not int(first) >= int(second):\n```\nThe `int()` casts are causing issues, as seen in the test case:\n```python\nassert_greater_equal(\"bar\", \"foo\")\n```\nwhich raises:\n```\nValueError: invalid literal for int() with base 10: \"bar\"\n```\n\n### Commit Summary:\nTo address this issue, the commit replaces the integer casting with a direct comparison. This ensures that the function works correctly with various data types that support comparison without unnecessary and erroneous casting to integers.\n\n### Commit Details:\n```python\ndef assert_greater_equal(first, second, msg=None):\n    \"\"\"Fail if `first` is not greater than or equal to `second`.\"\"\"\n    if not first >= second:\n        fail(msg or f\"{first!r} is not greater than or equal to {second!r}\")\n```\n### How the Commit Solves the Issue:\n1. **Direct Comparison**:\n   - The problematic `int(first) >= int(second)` is replaced with `first >= second`.\n   - This change eliminates the `ValueError` for types like strings or floats that do not require type casting to `int` for a valid comparison.\n\n2. **Type-Agnostic**:\n   - The revised comparison works correctly for any data type that supports the `>=` operator. This includes integers, floats, strings, and custom objects implementing comparison methods, enhancing the function's flexibility and robustness.\n   \n3. **Maintaining Functionality**:\n   - The core functionality of the assertion, ensuring `first` is greater than or equal to `second`, remains intact. The fix only alters the incorrect casting behavior without changing the logical operation.\n\n### Detailed Explanation for Developers:\nWhen encountering the `assert_greater_equal` function, developers realized that the use of `int()` was too restrictive and not type-compatible with various argument types. By debugging and reviewing the `ValueError` logs, it became apparent that this casting was causing more harm by introducing errors where valid comparisons should have succeeded.\n\nBy removing the `int` cast, the comparison is now performed directly:\n```python\nif not first >= second:\n```\nThis straightforward check respects the natural ordering rules defined for the types being compared. For instance:\n- **Strings**: `\"bar\" >= \"foo\"` correctly evaluates to `False`.\n- **Floats**: `2.5 >= 2.1` correctly evaluates to `True`.\n- **Custom Types**: Any objects with appropriately defined `__ge__` methods (greater than or equal to operator).\n\nThis approach maintains the function’s breadth of use while avoiding unnecessary and error-prone type transformations. The result is a more robust and user-friendly assertion method that behaves as expected across a broader range of input types, solving the initial logical error effectively."
    },
    "issue_ground_truth": {
      "title": "Logical Error in assert_greater_equal Comparison",
      "description": "There is an issue in the `assert_greater_equal` function where the comparison between the first and second arguments is incorrectly cast to integers using `int()`. This can cause unexpected behavior when dealing with non-integer types, such as floats or strings, leading to incorrect assertion results. Users may observe assertion failures even when the first argument is indeed greater than or equal to the second argument, but type casting changes the outcome. This needs to be fixed to ensure proper functionality and accuracy of the `assert_greater_equal` function without unwanted type casting.",
      "explanation": "### Summary of the Issue and Solution\n\n#### Summary of the Issue:\nThe issue reported involves a logical error in the `assert_greater_equal` function. The function compares two arguments to determine if the first is greater than or equal to the second. However, the function incorrectly casts both arguments to integers before performing the comparison. This typecasting can lead to unexpected results when the arguments are not integers, such as when they are floats or strings. Consequently, valid comparisons might fail, resulting in incorrect assertion results. This issue affects the robustness and reliability of the `assert_greater_equal` function.\n\n#### Details of the Commit:\nThe commit provides a fix for the `assert_greater_equal` function by removing the unnecessary integer typecasting. The commit changes the function to perform a direct comparison of the original arguments without altering their types.\n\n#### Explanation of the Cause of the Issue:\nThe issue arises because the `assert_greater_equal` function forces both arguments into integer type before performing the comparison. This type coercion can alter the values of the arguments when they are not naturally integers, such as floats, strings, or other comparable types. For instance, the comparison between \"3\" (string) and 2 (integer) after converting both to integers yields a comparison between 3 and 2, which might still seem valid. However, comparing 2.9 (float) and 2 by casting them both to integers before the comparison would inaccurately reduce the float to 2, leading to an incorrect result.\n\n#### Solution Explained:\nThe commit addresses the issue by simplifying the comparison logic in `assert_greater_equal`. Instead of converting the arguments to integers, it retains their original types and compares them directly. This ensures that the function works correctly for any comparable types, such as floats or strings, without altering their values.\n\n### How the Commit Solves the Issue:\nBy removing the integer cast and comparing the arguments directly, the commit ensures that `assert_greater_equal` correctly reflects the comparison result of the original input types. This preserves the natural ordering and behavior of the types being compared.\n\n### Detailed Explanation of the Solution:\nTo better understand, let's break down the key improvements brought by this commit:\n\n1. **Type Integrity**: The commit ensures that the function respects the original data types of the arguments. This prevents unintended data loss or alteration caused by casting to integers.\n\n2. **Comprehensive Validation**: The updated function now correctly handles various types such as integers, floats, strings, or any other types that support comparison operations. This flexibility improves the utility and correctness of the function.\n\n3. **Improved Robustness**: By performing direct comparisons, the function avoids potential pitfalls associated with typecasting, such as truncating decimal values or changing string properties, which could previously lead to unexpected behavior or false assertion failures.\n\n### Conclusion:\nThe issue with `assert_greater_equal` was caused by inappropriate integer typecasting of arguments before comparison. This was resolved by the commit, which removes the unnecessary typecasting and allows direct comparison of the original values. This change makes the function more robust and reliable, capable of correctly handling a broader range of input types without introducing unexpected behavior."
    },
    "location_origin": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "47": "assert_true"
        },
        "content_all": {
          "44": "    raise AssertionError(msg or \"assertion failure\")\n",
          "45": "\n",
          "46": "\n",
          "47": "def assert_true(expr, msg_fmt=\"{msg}\"):\n",
          "48": "    \"\"\"Fail the test unless the expression is truthy.\n",
          "49": "\n",
          "50": "    >>> assert_true(\"Hello World!\")\n",
          "51": "    >>> assert_true(\"\")\n",
          "52": "    Traceback (most recent call last):\n",
          "53": "        ...\n",
          "54": "    AssertionError: '' is not truthy\n",
          "55": "\n",
          "56": "    The following msg_fmt arguments are supported:\n",
          "57": "    * msg - the default error message\n",
          "58": "    * expr - tested e(...truncated)\n",
          "59": "    \"\"\"\n",
          "60": "    if not expr:\n",
          "61": "        raise AssertionError(msg_fmt.format(\n",
          "62": "            msg=f\"'{expr}' is not truthy\",\n",
          "63": "            expr=expr,\n",
          "64": "            additional_info=\"\",\n",
          "65": "            error_code=\"\"\n",
          "66": "        ))\n",
          "67": "\n",
          "68": "# Additional lines as required...\n"
        },
        "content_change": {
          "46": "def assert_true(expr, msg_fmt=\"{msg}\"):\n",
          "47": "    \"\"\"Fail the test unless the expression is truthy.\n",
          "48": "    >>> assert_true(\"Hello World!\")\n",
          "49": "    >>> assert_true(\"\")\n",
          "50": "    Traceback (most recent call last):\n",
          "51": "        ...\n",
          "52": "    AssertionError: '' is not truthy\n",
          "53": "    The following msg_fmt arguments are supported:\n",
          "54": "    * msg - the default error message\n",
          "55": "    * expr - tested expression that should be truthy\n",
          "56": "    * additional_info - extra context provided if needed\n",
          "57": "    * error_code - potential error code related to the assertion failure\n",
          "58": "    \"\"\"\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "529": "assert_greater_equal"
        },
        "content_all": {
          "526": "    if not int(first) >= int(second):",
          "527": "        fail(f\"Expected {first} >= {second}\")",
          "528": "\n",
          "529": "def assert_greater_equal(first, second):",
          "530": "    if not int(first) >= int(second):",
          "531": "        fail(f\"Expected {first} >= {second}\")",
          "532": "\n",
          "533": "def assert_has_attr(obj, name):",
          "534": "    \"\"\"Fail if the object does not have the given attribute.\n"
        },
        "content_change": {
          "530": "    try:",
          "531": "        if isinstance(first, (int, float)) and isinstance(second, (int, float)):",
          "532": "            if not first >= second:",
          "533": "                fail(f\"Expected {first} >= {second}\")",
          "534": "        else:",
          "535": "            try:",
          "536": "                if not float(first) >= float(second):",
          "537": "                    fail(f\"Expected {first} >= {second}\")",
          "538": "            except ValueError:",
          "539": "                if not str(first) >= str(second):",
          "540": "                    fail(f\"Expected {first} >= {second}\")",
          "541": "    except TypeError as e:",
          "542": "        fail(f\"Comparison failed: {e}\")"
        }
      }
    ],
    "location_ground": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "529": "assert_greater_equal"
        },
        "content_all": {
          "526": "    \"\"\"Fail if `first` is not greater than or equal to `second`.\"\"\"\n",
          "527": "\n",
          "528": "    if not int(first) >= int(second):\n",
          "529": "        fail(msg or f\"{first!r} is not greater than or equal to {second!r}\")\n",
          "530": "\n",
          "531": "\n",
          "532": "def assert_less_equal(first, second, msg=None):\n"
        },
        "content_change": {
          "528": "    if not first >= second:\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "110": "assert_greater_equal"
        },
        "content_all": {
          "107": "\n",
          "108": "def assert_greater_equal(a, b, msg_fmt=\"{a} is not greater than or equal to {b}\"):\n",
          "109": "    \"\"\"Fail if a is not greater than or equal to b.\"\"\"\n",
          "110": "    if not int(a) >= int(b):\n",
          "111": "        fail(msg_fmt.format(a=a, b=b, msg=f\"{a} is not greater than or equal to {b}\"))\n",
          "112": "    \n",
          "113": "\n"
        },
        "content_change": {
          "110": "    if not a >= b:\n"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "513": "assert_greater_equal"
        },
        "content_all": {
          "526": "    * second - the second argument\n",
          "527": "    \"\"\"\n",
          "528": "\n",
          "529": "    if not int(first) >= int(second):\n",
          "530": "        msg = \"{!r} is not greater than or equal to {!r}\".format(first, second)\n"
        },
        "content_change": {
          "529": "    if not int(first) >= int(second):\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": null,
        "content_all": {
          "531": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
          "532": "\n"
        },
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -54,7 +54,11 @@\n      AssertionError: '' is not truthy\n\n      The following msg_fmt arguments are supported:\n-     * msg - the default error message\n-     * expr - tested e(...truncated)\n+     * msg - The default error message.\n+     * expr - The tested expression that should be truthy.\n+     * additional_info - Extra context provided if needed.\n+     * error_code - Potential error code related to the assertion failure.\n+     \"\"\"\n     if not expr:\n         raise AssertionError(msg_fmt.format(\n             msg=f\"'{expr}' is not truthy\",\n             expr=expr,\n             additional_info=\"\",\n             error_code=\"\"\n         ))\n",
    "patch_im": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -47,6 +47,23 @@\n     \"\"\"Fail the test unless the expression is truthy.\n \n     >>> assert_true(\"Hello World!\")\n@@ -250,6 +267,27 @@\n     fail(\"Expected {} to be greater than {}\".format(a, b))\n \n \n+def assert_greater_equal(first, second):\n+    try:\n+        if isinstance(first, (int, float)) and isinstance(second, (int, float)):\n+            # If both are numbers, we compare directly\n+            if not first >= second:\n+                fail(f\"Expected {first} >= {second}\")\n+        else:\n+            # Attempt to convert to float and compare for numeric strings\n+            try:\n+                if not float(first) >= float(second):\n+                    fail(f\"Expected {first} >= {second}\")\n+            except ValueError:\n+                # Direct comparison for non-numeric strings or other types\n+                if not str(first) >= str(second):\n+                    fail(f\"Expected {first} >= {second}\")\n+    except TypeError as e:\n+        fail(f\"Comparison failed: {e}\")\n+\n+\n def assert_has_attr(obj, attr, msg_fmt=\"{msg}\"):\n     \"\"\"Fail the test if the object *obj* does not have the attribute *attr*.\n \n--- a/srittau_python-asserts/test_asserts.py\n+++ b/srittau_python-asserts/test_asserts.py\n@@ -88,6 +88,18 @@\n         assert_almost_equal(0.123456789, 0.123456780, places=6)\n \n     def test_assert_greater_equal(self):\n+        assert_greater_equal(10, 5)\n+        assert_greater_equal(5, 5)\n+        assert_greater_equal(3.14, 2.71)\n+        assert_greater_equal(\"foo\", \"bar\")  # Valid string comparison case\n+        try:\n+            assert_greater_equal(\"bar\", \"foo\")\n+        except AssertionError as e:\n+            assert str(e) == \"Expected bar >= foo\"\n+        try:\n+            assert_greater_equal(\"bar\", 42)\n+        except AssertionError as e:\n+            assert str(e) == \"Comparison failed: '>=' not supported between instances of 'str' and 'int'\"\n         assert_raises(AssertionError, assert_greater_equal, 1, 2)\n         assert_raises(AssertionError, assert_greater_equal, 1, 1, fail_msg=\"Expected 1 > 1\")\n         \n",
    "patch_il": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -44,13 +44,20 @@\n     raise AssertionError(msg or \"assertion failure\")\n \n \n def assert_true(expr, msg_fmt=\"{msg}\"):\n     \"\"\"Fail the test unless the expression is truthy.\n\n     >>> assert_true(\"Hello World!\")\n     >>> assert_true(\"\")\n     Traceback (most recent call last):\n         ...\n     AssertionError: '' is not truthy\n\n-    The following msg_fmt arguments are supported:\n-    * msg - the default error message\n-    * expr - tested e(...truncated)\n+    The following msg_fmt arguments are supported:\n+    * msg - the default error message\n+    * expr - the tested expression that should be truthy\n+    * additional_info - extra context provided if needed\n+    * error_code - potential error code related to the assertion failure\n     \"\"\"\n     if not expr:\n         raise AssertionError(msg_fmt.format(\n             msg=f\"'{expr}' is not truthy\",\n             expr=expr,\n             additional_info=\"\",\n             error_code=\"\"\n",
    "patch_iml": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -526,7 +526,18 @@\n \n def assert_greater_equal(first, second):\n-    if not int(first) >= int(second):\n-        fail(f\"Expected {first} >= {second}\")\n+    try:\n+        if isinstance(first, (int, float)) and isinstance(second, (int, float)):\n+            if not first >= second:\n+                fail(f\"Expected {first} >= {second}\")\n+        else:\n+            try:\n+                if not float(first) >= float(second):\n+                    fail(f\"Expected {first} >= {second}\")\n+            except ValueError:\n+                if not str(first) >= str(second):\n+                    fail(f\"Expected {first} >= {second}\")\n+    except TypeError as e:\n+        fail(f\"Comparison failed: {e}\")\n\n",
    "patch_ground": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -526,7 +526,7 @@ def assert_not_almost_equal(first, second, places=7, msg_fmt=\"{msg}\"):\n \n def assert_greater_equal(first, second, msg=None):\n     \"\"\"Fail if `first` is not greater than or equal to `second`.\"\"\"\n-    if not int(first) >= int(second):\n+    if not first >= second:\n         fail(msg or f\"{first!r} is not greater than or equal to {second!r}\")\n",
    "patch_ground_location": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -525,7 +525,7 @@ def assert_true(expr, msg_fmt=\"{msg}\"):\n \n \n def assert_greater_equal(first, second, msg=None):\n-    if not int(first) >= int(second):\n+    if not first >= second:\n         fail(msg or f\"{first!r} is not greater than or equal to {second!r}\")\n \n \n",
    "patch_ground_exp": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -107,8 +107,8 @@\n \n def assert_greater_equal(a, b, msg_fmt=\"{a} is not greater than or equal to {b}\"):\n     \"\"\"Fail if a is not greater than or equal to b.\"\"\"\n-    if not int(a) >= int(b):\n-        fail(msg_fmt.format(a=a, b=b, msg=f\"{a} is not greater than or equal to {b}\"))\n+    if not a >= b:\n+        fail(msg_fmt.format(a=a, b=b, msg=f\"{a} is not greater than or equal to {b}\"))\n \n",
    "patch_ground_all": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -526,7 +526,7 @@\n     * second - the second argument\n     \"\"\"\n \n-    if not int(first) >= int(second):\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",
    "patch_ground_truth": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -526,7 +526,7 @@\n     * second - the second argument\n     \"\"\"\n \n-    if not int(first) >= int(second):\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",
    "message": "======================================================================\nERROR: test_assert_greater_equal (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 579, in test_assert_greater_equal\n    assert_greater_equal(\"bar\", \"foo\")\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 529, in assert_greater_equal\n    if not int(first) >= int(second):\nValueError: invalid literal for int() with base 10: \"bar\"",
    "CodeBase": [
      {
        "path": "srittau_python-asserts/asserts/__init__.py",
        "content": "1 \"\"\"\n2 Rich Assertions.\n3 \n4 This module contains several rich standard assertions that can be used in unit\n5 tests and in implementations. Users are encouraged to define their own\n6 assertions, possibly using assertions from this package as a basis.\n7 \n8     >>> assert_equal(13, 13)\n9     >>> assert_equal(13, 14)\n10     Traceback (most recent call last):\n11         ...\n12     AssertionError: 13 != 14\n13     >>> with assert_raises(KeyError):\n14     ...     raise KeyError()\n15     >>> with assert_raises(KeyError):\n16     ...     pass\n17     Traceback (most recent call last):\n18         ...\n19     AssertionError: KeyError not raised\n20 \n21 \"\"\"\n22 \n23 from __future__ import annotations\n24 \n25 import re\n26 import sys\n27 from datetime import datetime, timedelta, timezone\n28 from json import loads as json_loads\n29 from typing import Any, Callable, Set\n30 from warnings import WarningMessage, catch_warnings\n31 \n32 from typing_extensions import deprecated\n33 \n34 \n35 def fail(msg=None):\n36     \"\"\"Raise an AssertionError with the given message.\n37 \n38     >>> fail(\"my message\")\n39     Traceback (most recent call last):\n40         ...\n41     AssertionError: my message\n42 \n43     \"\"\"\n44     raise AssertionError(msg or \"assertion failure\")\n45 \n46 \n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested e(...truncated)"
      },
      {
        "path": "srittau_python-asserts/test_asserts.py",
        "content": "1 # -*- coding: utf-8 -*-\n2 \n3 import re\n4 import sys\n5 from collections import OrderedDict\n6 from datetime import datetime, timedelta, timezone\n7 from json import JSONDecodeError\n8 from unittest import TestCase\n9 from warnings import catch_warnings, simplefilter, warn\n10 \n11 from asserts import (\n12     Absent,\n13     Exists,\n14     Present,\n15     assert_almost_equal,\n16     assert_between,\n17     assert_boolean_false,\n18     assert_boolean_true,\n19     assert_count_equal,\n20     assert_datetime_about_now,\n21     assert_datetime_about_now_utc,\n22     assert_dict_equal,\n23     assert_dict_superset,\n24     assert_equal,\n25     assert_false,\n26     assert_greater,\n27     assert_greater_equal,\n28     assert_has_attr,\n29     assert_in,\n30     assert_is,\n31     assert_is_instance,\n32     assert_is_none,\n33     assert_is_not,\n34     assert_is_not_none,\n35     assert_json_subset,\n36     assert_less,\n37     assert_less_equal,\n38     assert_not_almost_equal,\n39     assert_not_equal,\n40     assert_not_in,\n41     assert_not_is_instance,\n42     assert_not_regex,\n43     assert_raises,\n44     assert_raises_errno,\n45     assert_raises_regex,\n46     assert_regex,\n47     assert_succee(...truncated)"
      },
      {
        "path": "srittau_python-asserts/CHANGELOG.md",
        "content": "1 # Changelog for python-asserts\n2 \n3 python-asserts adheres to [semantic versioning](https://semver.org/).\n4 \n5 ## UNRELEASED –\n6 \n7 ## [0.13.1] – 2024-04-29\n8 \n9 ### Fixed\n10 \n11 Fixed Python 3.12 deprecation warnings.\n12 \n13 ## [0.13.0] – 2024-03-13\n14 \n15 ### Added\n16 \n17 - Add support for Python 3.12.\n18 - Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n19 \n20 ### Removed\n21 \n22 - Drop support for Python 3.7.\n23 \n24 ### Deprecated\n25 \n26 - Deprecate `Exists` in favor of `Present` and `Absent` in\n27   `assert_json_subset()`.\n28 \n29 ## [0.12.0]\n30 \n31 ### Added\n32 \n33 - Add `assert_not_regex()`.\n34 \n35 ### Changed\n36 \n37 - Modernize the type stubs.\n38 \n39 ### Removed\n40 \n41 - Drop support for Python 3.6.\n42 \n43 ## [0.11.1]\n44 \n45 ### Added\n46 \n47 - `assert_json_subset()` can now check for the existence or non-existence\n48   of object members using(...truncated)"
      },
      {
        "path": "srittau_python-asserts/README.md",
        "content": "1 # Python Asserts\n2 \n3 [![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n4 [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n5 [![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n6 [![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n7 (...truncated)"
      }
    ],
    "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 6,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 9,
      "Overall": 8
    },
    "issue_ground": {
      "Title": 8,
      "Description": 7,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Logical Error in assert_greater_equal Comparison",
      "description": "There is an issue in the `assert_greater_equal` function where the comparison between the first and second arguments is incorrectly cast to integers using `int()`. This can cause unexpected behavior when dealing with non-integer types, such as floats or strings, leading to incorrect assertion results. Users may observe assertion failures even when the first argument is indeed greater than or equal to the second argument, but type casting changes the outcome. This needs to be fixed to ensure proper functionality and accuracy of the `assert_greater_equal` function without unwanted type casting.",
      "explanation": "### Summary of the Issue and Solution\n\n#### Summary of the Issue:\nThe issue reported involves a logical error in the `assert_greater_equal` function. The function compares two arguments to determine if the first is greater than or equal to the second. However, the function incorrectly casts both arguments to integers before performing the comparison. This typecasting can lead to unexpected results when the arguments are not integers, such as when they are floats or strings. Consequently, valid comparisons might fail, resulting in incorrect assertion results. This issue affects the robustness and reliability of the `assert_greater_equal` function.\n\n#### Details of the Commit:\nThe commit provides a fix for the `assert_greater_equal` function by removing the unnecessary integer typecasting. The commit changes the function to perform a direct comparison of the original arguments without altering their types.\n\n#### Explanation of the Cause of the Issue:\nThe issue arises because the `assert_greater_equal` function forces both arguments into integer type before performing the comparison. This type coercion can alter the values of the arguments when they are not naturally integers, such as floats, strings, or other comparable types. For instance, the comparison between \"3\" (string) and 2 (integer) after converting both to integers yields a comparison between 3 and 2, which might still seem valid. However, comparing 2.9 (float) and 2 by casting them both to integers before the comparison would inaccurately reduce the float to 2, leading to an incorrect result.\n\n#### Solution Explained:\nThe commit addresses the issue by simplifying the comparison logic in `assert_greater_equal`. Instead of converting the arguments to integers, it retains their original types and compares them directly. This ensures that the function works correctly for any comparable types, such as floats or strings, without altering their values.\n\n### How the Commit Solves the Issue:\nBy removing the integer cast and comparing the arguments directly, the commit ensures that `assert_greater_equal` correctly reflects the comparison result of the original input types. This preserves the natural ordering and behavior of the types being compared.\n\n### Detailed Explanation of the Solution:\nTo better understand, let's break down the key improvements brought by this commit:\n\n1. **Type Integrity**: The commit ensures that the function respects the original data types of the arguments. This prevents unintended data loss or alteration caused by casting to integers.\n\n2. **Comprehensive Validation**: The updated function now correctly handles various types such as integers, floats, strings, or any other types that support comparison operations. This flexibility improves the utility and correctness of the function.\n\n3. **Improved Robustness**: By performing direct comparisons, the function avoids potential pitfalls associated with typecasting, such as truncating decimal values or changing string properties, which could previously lead to unexpected behavior or false assertion failures.\n\n### Conclusion:\nThe issue with `assert_greater_equal` was caused by inappropriate integer typecasting of arguments before comparison. This was resolved by the commit, which removes the unnecessary typecasting and allows direct comparison of the original values. This change makes the function more robust and reliable, capable of correctly handling a broader range of input types without introducing unexpected behavior."
    }
  }
}