{
  "RepoName": "https://github.com/srittau/python-asserts.git",
  "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4",
  "Type": "logic error",
  "ErrorMessage": "======================================================================\nFAIL: test_assert_json_subset__objects_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 1389, in test_assert_json_subset__objects_equal\n    assert_json_subset(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1403, in assert_\n    self._assert_dicts_equal()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1389, in test_assert_json_subset__objects_equal\n    assert_json_subset(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1152, in __exit__\n    fail(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: AssertionError was unexpectedly raised\n\n======================================================================\nFAIL: test_assert_json_subset__one_key_missing_from_first_object (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 1395, in test_assert_json_subset__one_key_missing_from_first_object\n    assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1403, in assert_\n    self._assert_dicts_equal()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1395, in test_assert_json_subset__one_key_missing_from_first_object\n    assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1152, in __exit__\n    fail(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: AssertionError was unexpectedly raised\n\n======================================================================\nFAIL: test_assert_json_subset__presence_check (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 1496, in test_assert_json_subset__presence_check\n    assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1403, in assert_\n    self._assert_dicts_equal()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1496, in test_assert_json_subset__presence_check\n    assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1152, in __exit__\n    fail(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: AssertionError was unexpectedly raised\n\n======================================================================\nFAIL: test_assert_json_subset__value_differs (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 1411, in test_assert_json_subset__value_differs\n    assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1411, in test_assert_json_subset__value_differs\n    assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 86, in __exit__\n    raise AssertionError(\nAssertionError: expected exception message \"element $[\"foo\"] differs: 3 != 4\", got \"spurious member \"foo\" in object $\"",
  "Issue": {
    "title": "Incorrect Spurious Member Check in JSON Subsets",
    "description": "There is an issue with the `_assert_no_wrong_keys` method in the `_JSONComparer` class from the `asserts` module. When checking for spurious members in JSON subsets, the logic incorrectly uses 'or' instead of `and` for conditionally determining if a key should be considered spurious. This can result in incorrect assertion failures for JSON objects being checked against expected subsets. The impact is significant because it affects the accuracy of JSON subset validations, which are common in tests checking for specific key presence or absence in structured data objects.\n\nA correct implementation should check whether `name` is a string **and** `_is_absent(self._expected[name])`. Fixing this bug ensures that false positives in identifying spurious keys are eliminated, thereby improving the reliability of JSON assertions in test cases.",
    "explanation": "### Summary of the Issue\n\nThe reported issue is related to a bug in the `_assert_no_wrong_keys` method within the `_JSONComparer` class in the `asserts` module. This method is responsible for validating JSON subsets to ensure the correctness of keys, specifically checking for spurious (unwanted or unexpected) keys in a JSON object.\n\nThe problem is that the logic incorrectly uses the logical 'or' condition instead of the logical 'and' condition when determining if a key should be considered spurious. This incorrect condition can lead to false positives, causing unnecessary assertion failures when validating JSON objects against expected subsets. The anticipated correction involves changing the condition to improve the accuracy of JSON subset validation, thereby enhancing the reliability of JSON assertions in tests.\n\n### Details of the Commit\n\nThe commit in question addresses this issue through the following modifications:\n\n- **Commit Message:** Not provided explicitly as a message but inferred from the context to relate it to fixing the logical condition in _assert_no_wrong_keys.\n- **File Affected:** `srittau_python-asserts/asserts/__init__.py`\n- **Changes Made:** The commit modifies the logical condition used in the `_assert_no_wrong_keys` method to correctly validate if a key is spurious.\n\n### Explanation of the Issue and Solution\n\n**Cause of the Issue:**\n\nThe `_assert_no_wrong_keys` method in the `_JSONComparer` class is designed to detect spurious keys in a JSON object when validating against an expected subset. This method checks if an expected key's name is a string or if the value is considered absent (using the `Absent` helper class).\n\nHowever, the original implementation uses an 'or' operator in the conditional check:\n```\nif isinstance(name, str) or _is_absent(self._expected[name]):\n```\nThis condition would incorrectly trigger a spurious key check if either part of the condition is true:\n- **isinstance(name, str):** The key is a string.\n- **_is_absent(self._expected[name]):** The value associated with the key in the expected JSON subset is absent.\n\nThis logical 'or' means that the condition will trigger if either the key is a string, or the value is absent, which is not always desired. It could lead to incorrect assertions for JSON objects that have valid keys present in the subset due to the loose condition.\n\n**Solution from the Commit:**\n\nThe correction involves changing this logical 'or' to a logical 'and':\n```\nif isinstance(name, str) and _is_absent(self._expected[name]):\n```\nThis adjustment ensures that the spurious member error is raised only if:\n- The key is a string and its value is marked as absent in the expected subset.\n\nBy tightening the logical condition to enforce both clauses to be true simultaneously, it accurately checks for unwanted keys that should not be present as per the expected subset.\n\n### How the Commit Solves the Issue:\n\n1. **Correct Logic:** The revised logical condition using 'and' ensures the method only flags keys marked as absent, improving the accuracy of the spurious key check and avoiding false positives.\n2. **Accurate Validation:** With the correct condition in place, validation of JSON subsets becomes more reliable. Only the actual spurious keys (keys that should be absent) are flagged, enhancing the robustness of the assertions.\n3. **Improved Confidence:** Developers can have better confidence in their tests, as the validation mechanism now aligns more closely with the intended design, checking both the type and the presence accurately.\n\n### Final Explanation:\n\nIn summary, the issue was due to an incorrect logical condition in the `_assert_no_wrong_keys` method, where an 'or' condition led to incorrect identification of spurious keys. The commit addressed this by changing the condition to 'and', ensuring both criteria must be met to flag a key as spurious accurately. This correction improves the reliability and correctness of JSON subset validations in the testing framework, ultimately reducing false positives and enhancing overall test accuracy."
  },
  "Explain": "### Summary of the Issue\n\nThe reported issue is related to a bug in the `_assert_no_wrong_keys` method within the `_JSONComparer` class in the `asserts` module. This method is responsible for validating JSON subsets to ensure the correctness of keys, specifically checking for spurious (unwanted or unexpected) keys in a JSON object.\n\nThe problem is that the logic incorrectly uses the logical 'or' condition instead of the logical 'and' condition when determining if a key should be considered spurious. This incorrect condition can lead to false positives, causing unnecessary assertion failures when validating JSON objects against expected subsets. The anticipated correction involves changing the condition to improve the accuracy of JSON subset validation, thereby enhancing the reliability of JSON assertions in tests.\n\n### Details of the Commit\n\nThe commit in question addresses this issue through the following modifications:\n\n- **Commit Message:** Not provided explicitly as a message but inferred from the context to relate it to fixing the logical condition in _assert_no_wrong_keys.\n- **File Affected:** `srittau_python-asserts/asserts/__init__.py`\n- **Changes Made:** The commit modifies the logical condition used in the `_assert_no_wrong_keys` method to correctly validate if a key is spurious.\n\n### Explanation of the Issue and Solution\n\n**Cause of the Issue:**\n\nThe `_assert_no_wrong_keys` method in the `_JSONComparer` class is designed to detect spurious keys in a JSON object when validating against an expected subset. This method checks if an expected key's name is a string or if the value is considered absent (using the `Absent` helper class).\n\nHowever, the original implementation uses an 'or' operator in the conditional check:\n```\nif isinstance(name, str) or _is_absent(self._expected[name]):\n```\nThis condition would incorrectly trigger a spurious key check if either part of the condition is true:\n- **isinstance(name, str):** The key is a string.\n- **_is_absent(self._expected[name]):** The value associated with the key in the expected JSON subset is absent.\n\nThis logical 'or' means that the condition will trigger if either the key is a string, or the value is absent, which is not always desired. It could lead to incorrect assertions for JSON objects that have valid keys present in the subset due to the loose condition.\n\n**Solution from the Commit:**\n\nThe correction involves changing this logical 'or' to a logical 'and':\n```\nif isinstance(name, str) and _is_absent(self._expected[name]):\n```\nThis adjustment ensures that the spurious member error is raised only if:\n- The key is a string and its value is marked as absent in the expected subset.\n\nBy tightening the logical condition to enforce both clauses to be true simultaneously, it accurately checks for unwanted keys that should not be present as per the expected subset.\n\n### How the Commit Solves the Issue:\n\n1. **Correct Logic:** The revised logical condition using 'and' ensures the method only flags keys marked as absent, improving the accuracy of the spurious key check and avoiding false positives.\n2. **Accurate Validation:** With the correct condition in place, validation of JSON subsets becomes more reliable. Only the actual spurious keys (keys that should be absent) are flagged, enhancing the robustness of the assertions.\n3. **Improved Confidence:** Developers can have better confidence in their tests, as the validation mechanism now aligns more closely with the intended design, checking both the type and the presence accurately.\n\n### Final Explanation:\n\nIn summary, the issue was due to an incorrect logical condition in the `_assert_no_wrong_keys` method, where an 'or' condition led to incorrect identification of spurious keys. The commit addressed this by changing the condition to 'and', ensuring both criteria must be met to flag a key as spurious accurately. This correction improves the reliability and correctness of JSON subset validations in the testing framework, ultimately reducing false positives and enhancing overall test accuracy.",
  "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 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) or _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@@ -1442,7 +1442,7 @@\n \n     def _assert_no_wrong_keys(self) -> None:\n         for name in self._expected:\n-            if isinstance(name, str) or _is_absent(self._expected[name]):\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",
  "BuggyCodeLocation": [
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": {
        "1443": "_assert_no_wrong_keys"
      },
      "content_all": {
        "1442": "\n",
        "1443": "    def _assert_no_wrong_keys(self) -> None:\n",
        "1444": "        for name in self._expected:\n",
        "1445": "            if isinstance(name, str) or _is_absent(self._expected[name]):\n",
        "1446": "                if name in self._actual:\n",
        "1447": "                    self._raise_assertion_error(\n",
        "1448": "                        f\"spurious member '{name}' in object {{path}}\"\n"
      },
      "content_change": {
        "1445": "            if isinstance(name, str) or _is_absent(self._expected[name]):\n"
      }
    },
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    }
  ],
  "Source": "Human",
  "Command": [
    "python -m unittest test_asserts.py"
  ],
  "Token": 1151,
  "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/asserts/__init__.pyi",
      "content": "1 import datetime\n2 from collections.abc import Callable, Container, Iterable\n3 from contextlib import AbstractContextManager as ContextManager\n4 from re import Pattern\n5 from types import TracebackType\n6 from typing import Any, Generic, NoReturn, TypeVar\n7 \n8 from typing_extensions import deprecated\n9 \n10 _E = TypeVar(\"_E\", bound=BaseException)\n11 _S = TypeVar(\"_S\")\n12 \n13 class AssertRaisesContext(Generic[_E]):\n14     exception: type[_E]\n15     msg_fmt: str\n16     def __init__(self, exc(...truncated)"
    }
  ],
  "TokenAll": 38742,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Deprecation of 'Exists' assertion",
      "description": "The 'Exists' assertion is deprecated in favor of 'Present' and 'Absent' in assert_json_subset() as mentioned in the changelog. Code still using the 'Exists' assertion should be updated to use the new assertions to ensure compatibility with future versions.",
      "explanation": "### Issue Summary\n\nThe issue at hand details the deprecation of the 'Exists' assertion. It states that the 'Exists' assertion has been deprecated and should be replaced by 'Present' and 'Absent' assertions within the `assert_json_subset()` function. This change is noted to ensure compatibility with future versions of the library.\n\n### Code Content Overview\n\nTo address this issue, we examine several files, including the main `__init__.py` file of the `asserts` module, the `test_asserts.py` file which contains test cases, the `CHANGELOG.md` file which documents historical updates, and the `__init__.pyi` for type stubs. The focal point for making updates should be the function `assert_json_subset()` found in `asserts/__init__.py`, wherever the `Exists` assertion is used.\n\n### Commit Analysis\n\nLet's break down the necessary changes in detail:\n\n#### 1. **Deprecate `Exists` in `asserts/__init__.py`**\n\n   - **Locate the use of `Exists` assertion**: The commit should locate any functionality or documentation within the code that still uses `Exists`.\n   - **Replace with `Present` and `Absent`**: Update those instances to use the new assertions `Present` and `Absent`.\n\n#### 2. **Update `test_asserts.py`**\n\n   - **Adjust Tests**: Tests using the `Exists` assertion should be modified to use `Present` or `Absent` for correctness and compatibility.\n   \n#### 3. **Update CHANGELOG.md**\n\n   - **Document Changes**: Ensure that the `CHANGELOG.md` properly reflects the deprecation of the `Exists` assertion and informs about the replacement with `Present` and `Absent`.\n\n### Implementation Details\n\nHere are how the changes might look in the files:\n\n#### Updates in `asserts/__init__.py`\n\nLocate and update the section dealing with JSON subset assertion:\n\n**Before:**\n```python\n11 from asserts import (\n12     Exists, \n    # other imports...\n)\n35 def assert_json_subset(expected, actual, message=None):\n    ...\n```\n\n**After:**\n```python\n11 from asserts import (\n12     Present, \n13     Absent, \n    # other imports...\n)\n35 def assert_json_subset(expected, actual, message=None):\n    ...\n```\n\n#### Updates in `test_asserts.py`\n\nReplace occurrences of `Exists` with `Present` or `Absent` in the test cases:\n\n**Before:**\n```python\n11 from asserts import (\n12     Exists,\n    # other imports...\n)\n\ndef test_json_subset():\n    assert_json_subset({\"key\": Exists}, {\"key\": \"value\"})\n```\n\n**After:**\n```python\n11 from asserts import (\n12     Present,\n13     Absent,\n    # other imports...\n)\n\ndef test_json_subset():\n    assert_json_subset({\"key\": Present}, {\"key\": \"value\"})\n```\n\n#### Update in `CHANGELOG.md`\n\nEnsure changelog reflects this change appropriately:\n\n**Existing Record:**\n```markdown\n26 - Deprecate `Exists` in favor of `Present` and `Absent` in\n27   `assert_json_subset()`.\n```\n\n**Additional Note or Extension:**\n```markdown\n26 - Deprecate `Exists` in favor of `Present` and `Absent` in\n27   `assert_json_subset()`. All instances of `Exists` should be replaced to ensure compatibility with future versions.\n```\n\n### Explanation of Solution\n\nThe commit effectively phases out the deprecated `Exists` assertion by:\n\n1. **Source Code Update**: Updating the imports and usage within the `asserts/__init__.py` to replace `Exists` with `Present` or `Absent`.\n   \n2. **Test Code Update**: Amending the test cases to replace `Exists` with `Present` or `Absent`, ensuring that all tests align with the new assertions and test coverage remains intact.\n\n3. **Documentation Update**: Reflecting the changes in imports and usage within the `CHANGELOG.md` which serves as a guide for developers to understand why `Exists` was deprecated and what their code should now utilize.\n\nThese changes ensure developers who use this library transition smoothly, maintaining compatibility and functionality as they upgrade to future versions of the `assert_json_subset()` function. By providing clear alternatives (`Present` and `Absent`), the commit anticipates and prevents future issues related to the deprecation of `Exists`."
    },
    "issue_message": {
      "title": "Incorrect error message in assert_json_subset",
      "description": "The `assert_json_subset` function in asserts/__init__.py fails with an incorrect error message 'spurious member \"foo\" in object $' instead of a more precise message. Adjust the function to provide correct and specific error messages.",
      "explanation": "### Summary of the Issue\n\nThe `assert_json_subset` function in the `asserts/__init__.py` file is producing incorrect and non-specific error messages such as `'spurious member \"foo\" in object $'` when the JSON objects being compared differ. This vague error message makes it difficult to pinpoint the exact discrepancy between the objects. The expected behavior is to provide a more precise and informative error message highlighting the exact difference.\n\n### Analysis of the Code\n\n#### Relevant Part of the Code: `assert_json_subset`\n\nThe `assert_json_subset` function is designed to compare two JSON objects. It raises an AssertionError if the JSON object (subset) is not a subset of the other (superset). Here's a key excerpt from `asserts/__init__.py`:\n\n```python\ndef assert_json_subset(subset, superset, path=\"$\"):\n    comparer = JSONComparer(subset, superset, path)\n    comparer.assert_()\n```\n\n#### Error Trigger\n\nThe test error messages indicate that the function fails within `assert_json_subset`, specifically when calling the `assert_` method from the `JSONComparer` class. The relevant traceback messages indicate problems around lines 1389-1403 in `asserts/__init__.py`.\n\n### Commit Analysis\n\nTo fix the issue, consider a hypothetical commit as shown below. Let's assume that the commit message is `\"Fix incorrect error messages in assert_json_subset\".`\n\n```diff\ndiff --git a/asserts/__init__.py b/asserts/__init__.py\nindex abcdefg..hijklmn 100644\n--- a/asserts/__init__.py\n+++ b/asserts/__init__.py\n@@ -1387,19 +1387,29 @@ class JSONComparer:\n         self.superset = superset\n         self.path = path\n \n-    def assert_(self):\n-        self._assert_dicts_equal()\n+    def assert_(self) -> None:\n+        if isinstance(self.subset, dict) and isinstance(self.superset, dict):\n+            self._assert_dicts_equal()\n+        elif self.subset != self.superset:\n+            self._fail(\n+                \"element {} differs: {} != {}\".format(\n+                    self.path, self.subset, self.superset\n+                )\n+            )\n \n     def _assert_dicts_equal(self) -> None:\n         for key, value in self.subset.items():\n             if key not in self.superset:\n-                self._fail(f'spurious member \"{key}\" in object {self.path}')\n+                self._fail(f'missing member \"{key}\" in object {self.path}')\n             else:\n                 JSONComparer(value, self.superset[key], f'{self.path}[\"{key}\"]').assert_()\n \n     def _fail(self, message: str) -> None:\n-        raise AssertionError(message)\n+        fail(message)\n \n+def fail(msg=None):\n+    \"\"\"Raise an AssertionError with the given message.\"\"\"\n+    raise AssertionError(msg or \"assertion failure\")\n+\n def assert_json_subset(subset, superset, path=\"$\"):\n     comparer = JSONComparer(subset, superset, path)\n     comparer.assert_()\n ```\n\n### Explanation of the Fix\n\n1. **Enhanced Condition Handling**:\n   - The `assert_` method now accounts for both `dict` and non-dict (e.g., primitive types such as strings, integers) values. This broadens the function's ability to pinpoint more specific discrepancies between different JSON elements.\n\n2. **Precise Error Messages**:\n   - If a key is not found in the superset, the error now correctly states `'missing member \"{key}\" in object {self.path}'`, specifying the exact key and location.\n   - If the `subset` and `superset` values differ, it precisely states the path and the differing values with `'element {self.path} differs: {self.subset} != {self.superset}'`.\n\n3. **Refactored Fail Mechanism**:\n   - The `_fail` method now uses the `fail` function to consistently handle error throwing, streamlining error management.\n\n4. **Testing and Verification**:\n   - The comprehensive error message helps unit tests like `test_assert_json_subset__value_differs` or `test_assert_json_subset__presence_check` to pinpoint exact differences, providing immediate clarity on test failures.\n\n### How the Commit Solves the Issue\n\nThe changes ensure that `assert_json_subset` provides users with precise and context-specific error messages which reflect the exact mismatch or issue within the JSON structures being compared. Unit tests failing with these enhanced errors will now clearly explain the problem, facilitating easier debugging and validation.\n\nIn summary:\n- The fix correctly identifies whether elements are missing or different.\n- It ensures all discrepancies are communicated with specific paths and values, resolving the initial issue of vague error messages."
    },
    "issue_ground": {
      "title": "Incorrect Spurious Member Check in JSON Subsets",
      "description": "There is an issue with the `_assert_no_wrong_keys` method in the `_JSONComparer` class from the `asserts` module. When checking for spurious members in JSON subsets, the logic incorrectly uses 'or' instead of `and` for conditionally determining if a key should be considered spurious. This can result in incorrect assertion failures for JSON objects being checked against expected subsets. The impact is significant because it affects the accuracy of JSON subset validations, which are common in tests checking for specific key presence or absence in structured data objects.\n\nA correct implementation should check whether `name` is a string **and** `_is_absent(self._expected[name])`. Fixing this bug ensures that false positives in identifying spurious keys are eliminated, thereby improving the reliability of JSON assertions in test cases.",
      "explanation": "### Issue Summary\n\nThe issue involves the `_assert_no_wrong_keys` method in the `_JSONComparer` class within the `asserts` module. The method is responsible for identifying spurious members in JSON subsets. However, there's a logical error where an 'or' condition is used instead of 'and'. This error causes incorrect assertions, falsely flagging keys as spurious when validating JSON objects against expected subsets. The false positives disrupt the accuracy of tests that check the presence or absence of specific keys in JSON structures.\n\n### Analyzing the Code\n\nWithout the direct lines from the `_JSONComparer` class and `_assert_no_wrong_keys` method, we infer the situation from the error messages and change logs:\n```plaintext\nassertion failure\nspurious member \"foo\" in object $\n```\nThese indicate the method incorrectly identifies entries as spurious.\n\n### The Issue in Code\n\nWe need to correct the conditional logic used in `_assert_no_wrong_keys`.\n\nIncorrect original condition:\n```python\nif name not in self._expected or _is_absent(self._expected[name]):\n    # logic to handle spurious keys\n```\n\nThis condition fails because it uses 'or', which flags any missing key or present absent-marked key as spurious incorrectly. Correct logic requires both conditions.\n\n### Correct Implementation\n\nCorrect the condition to **both**:\n```python\nif name not in self._expected and _is_absent(self._expected[name]):\n    # logic to handle spurious keys\n```\n\n### How the Commit Fixes the Issue\n\n1. **Identifying the Condition:**\n   - Change the 'or' to 'and' ensures that a key is flagged as spurious only if:\n        - It is not in the expected JSON subset, **and**\n        - It has the `_is_absent` status in the expected subset — ensuring that keys explicitly marked as absent are correctly handled.\n\n2. **Fix Application:**\n   - **Fixed Code:**\n     ```python\n     def _assert_no_wrong_keys(self):\n         for name in self._actual:\n             if name not in self._expected and _is_absent(self._expected.get(name)):\n                 self._mark_as_error(f'spurious member \"{name}\" in object {self._path}')\n     ```\n\n### Explanation\n\n- **Previous Logic:**\n  - Any key missing from the expected subset or marked absent would trigger a false positive.\n\n- **New Logic:**\n  - `name not in self._expected and _is_absent(self._expected.get(name))` ensures:\n    - A key is considered spurious only if it is indeed not required (`not in self._expected`) and marked as absent, filtering out false positives.\n\n### Summary of Solution\n\nThe commit corrects the logic error by changing `or` to `and` in the spurious member check condition within `_assert_no_wrong_keys`. This change prevents incorrect assertion failures, restoring the integrity of JSON subset validation by ensuring keys flagged as absent are accurately verified only when truly necessary. This fix improves the accuracy and reliability of tests checking JSON structures for key presence or absence.\n\nBy making this change, all test cases that were incorrectly failing due to the spurious member flagging should now pass correctly, as it eliminates the false positives that were previously being generated due to the logical error in the condition."
    },
    "issue_ground_truth": {
      "title": "Incorrect Spurious Member Check in JSON Subsets",
      "description": "There is an issue with the `_assert_no_wrong_keys` method in the `_JSONComparer` class from the `asserts` module. When checking for spurious members in JSON subsets, the logic incorrectly uses 'or' instead of `and` for conditionally determining if a key should be considered spurious. This can result in incorrect assertion failures for JSON objects being checked against expected subsets. The impact is significant because it affects the accuracy of JSON subset validations, which are common in tests checking for specific key presence or absence in structured data objects.\n\nA correct implementation should check whether `name` is a string **and** `_is_absent(self._expected[name])`. Fixing this bug ensures that false positives in identifying spurious keys are eliminated, thereby improving the reliability of JSON assertions in test cases.",
      "explanation": "### Summary of the Issue\n\nThe reported issue is related to a bug in the `_assert_no_wrong_keys` method within the `_JSONComparer` class in the `asserts` module. This method is responsible for validating JSON subsets to ensure the correctness of keys, specifically checking for spurious (unwanted or unexpected) keys in a JSON object.\n\nThe problem is that the logic incorrectly uses the logical 'or' condition instead of the logical 'and' condition when determining if a key should be considered spurious. This incorrect condition can lead to false positives, causing unnecessary assertion failures when validating JSON objects against expected subsets. The anticipated correction involves changing the condition to improve the accuracy of JSON subset validation, thereby enhancing the reliability of JSON assertions in tests.\n\n### Details of the Commit\n\nThe commit in question addresses this issue through the following modifications:\n\n- **Commit Message:** Not provided explicitly as a message but inferred from the context to relate it to fixing the logical condition in _assert_no_wrong_keys.\n- **File Affected:** `srittau_python-asserts/asserts/__init__.py`\n- **Changes Made:** The commit modifies the logical condition used in the `_assert_no_wrong_keys` method to correctly validate if a key is spurious.\n\n### Explanation of the Issue and Solution\n\n**Cause of the Issue:**\n\nThe `_assert_no_wrong_keys` method in the `_JSONComparer` class is designed to detect spurious keys in a JSON object when validating against an expected subset. This method checks if an expected key's name is a string or if the value is considered absent (using the `Absent` helper class).\n\nHowever, the original implementation uses an 'or' operator in the conditional check:\n```\nif isinstance(name, str) or _is_absent(self._expected[name]):\n```\nThis condition would incorrectly trigger a spurious key check if either part of the condition is true:\n- **isinstance(name, str):** The key is a string.\n- **_is_absent(self._expected[name]):** The value associated with the key in the expected JSON subset is absent.\n\nThis logical 'or' means that the condition will trigger if either the key is a string, or the value is absent, which is not always desired. It could lead to incorrect assertions for JSON objects that have valid keys present in the subset due to the loose condition.\n\n**Solution from the Commit:**\n\nThe correction involves changing this logical 'or' to a logical 'and':\n```\nif isinstance(name, str) and _is_absent(self._expected[name]):\n```\nThis adjustment ensures that the spurious member error is raised only if:\n- The key is a string and its value is marked as absent in the expected subset.\n\nBy tightening the logical condition to enforce both clauses to be true simultaneously, it accurately checks for unwanted keys that should not be present as per the expected subset.\n\n### How the Commit Solves the Issue:\n\n1. **Correct Logic:** The revised logical condition using 'and' ensures the method only flags keys marked as absent, improving the accuracy of the spurious key check and avoiding false positives.\n2. **Accurate Validation:** With the correct condition in place, validation of JSON subsets becomes more reliable. Only the actual spurious keys (keys that should be absent) are flagged, enhancing the robustness of the assertions.\n3. **Improved Confidence:** Developers can have better confidence in their tests, as the validation mechanism now aligns more closely with the intended design, checking both the type and the presence accurately.\n\n### Final Explanation:\n\nIn summary, the issue was due to an incorrect logical condition in the `_assert_no_wrong_keys` method, where an 'or' condition led to incorrect identification of spurious keys. The commit addressed this by changing the condition to 'and', ensuring both criteria must be met to flag a key as spurious accurately. This correction improves the reliability and correctness of JSON subset validations in the testing framework, ultimately reducing false positives and enhancing overall test accuracy."
    },
    "location_origin": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "35": "assert_json_subset"
        },
        "content_all": {
          "32": "from typing_extensions import deprecated\n",
          "33": "\n",
          "34": "\n",
          "35": "def assert_json_subset(expected, actual, message=None):\n",
          "36": "    # Your existing implementation\n",
          "37": "    ...\n",
          "38": "from asserts import (\n",
          "39": "    Present, \n",
          "40": "    Absent, \n"
        },
        "content_change": {
          "38": "from asserts import (\n",
          "39": "    Present, \n",
          "40": "    Absent, \n"
        }
      },
      {
        "file": "srittau_python-asserts/test_asserts.py",
        "function": {
          "11": "test_json_subset"
        },
        "content_all": {
          "8": "from warnings import catch_warnings, simplefilter, warn\n",
          "9": "\n",
          "10": "from asserts import (\n",
          "11": "    Absent,\n",
          "12": "    Present,\n",
          "13": "    # other imports...\n",
          "14": ")\n",
          "15": "def test_json_subset():\n",
          "16": "    assert_json_subset({\"key\": Present}, {\"key\": \"value\"})\n"
        },
        "content_change": {
          "11": "Absent,\n",
          "12": "Present,\n"
        }
      },
      {
        "file": "srittau_python-asserts/CHANGELOG.md",
        "function": {
          "25": "Deprecation Notice"
        },
        "content_all": {
          "23": "\n",
          "24": "### Deprecated\n",
          "25": "- Deprecate `Exists` in favor of `Present` and `Absent` in\n",
          "26": "  `assert_json_subset()`. All instances of `Exists` should be replaced to ensure compatibility with future versions.\n",
          "27": "\n",
          "28": "## [0.12.0]\n",
          "29": "### Added\n",
          "30": "- Add `assert_not_regex()`.\n"
        },
        "content_change": {
          "25": "- Deprecate `Exists` in favor of `Present` and `Absent` in\n",
          "26": "  `assert_json_subset()`. All instances of `Exists` should be replaced to ensure compatibility with future versions.\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "1391": "assert_json_subset"
        },
        "content_all": {
          "1390": "def assert_json_subset(subset, superset, path=\"$\"):\n",
          "1391": "    comparer = JSONComparer(subset, superset, path)\n",
          "1392": "    comparer.assert_()\n",
          "1393": "\n",
          "1394": "class JSONComparer:\n",
          "1395": "    def __init__(self, subset, superset, path) -> None:\n",
          "1396": "        self.subset = subset\n"
        },
        "content_change": {
          "1391": "    comparer.assert_()\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "1403": "assert_"
        },
        "content_all": {
          "1400": "    def assert_(self) -> None:\n",
          "1401": "        if isinstance(self.subset, dict) and isinstance(self.superset, dict):\n",
          "1402": "            self._assert_dicts_equal()\n",
          "1403": "        elif self.subset != self.superset:\n",
          "1404": "            self._fail(\n",
          "1405": "                \"element {} differs: {} != {}\".format(\n",
          "1406": "                    self.path, self.subset, self.superset\n",
          "1407": "                )\n",
          "1408": "            )\n",
          "1409": "\n",
          "1410": "    def _assert_dicts_equal(self) -> None:\n",
          "1411": "        for key, value in self.subset.items():\n",
          "1412": "            if key not in self.superset:\n"
        },
        "content_change": {
          "1403": "        elif self.subset != self.superset:\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "1412": "_assert_dicts_equal"
        },
        "content_all": {
          "1411": "    def _assert_dicts_equal(self) -> None:\n",
          "1412": "        for key, value in self.subset.items():\n",
          "1413": "            if key not in self.superset:\n",
          "1414": "                self._fail(f'missing member \"{key}\" in object {self.path}')\n",
          "1415": "            else:\n",
          "1416": "                JSONComparer(value, self.superset[key], f'{self.path}[\"{key}\"]').assert_()\n",
          "1417": "\n",
          "1418": "    def _fail(self, message: str) -> None:\n",
          "1419": "        fail(message)\n",
          "1420": "\n",
          "1421": "def fail(msg=None):\n"
        },
        "content_change": {
          "1414": "                self._fail(f'missing member \"{key}\" in object {self.path}')\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "1419": "_fail"
        },
        "content_all": {
          "1418": "    def _fail(self, message: str) -> None:\n",
          "1419": "        fail(message)\n",
          "1420": "\n",
          "1421": "def fail(msg=None):\n",
          "1422": "    \"\"\"Raise an AssertionError with the given message.\"\"\"\n",
          "1423": "    raise AssertionError(msg or \"assertion failure\")\n",
          "1424": "\n",
          "1425": "def assert_true(expr, msg_fmt=\"{msg}\"):\n",
          "1426": "    \"\"\"Fail the test unless the expression is truthy.\n",
          "1427": "\n",
          "1428": "    >>> assert_true(\"Hello World!\")\n"
        },
        "content_change": {
          "1419": "        fail(message)\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "139": "_JSONComparer._assert_no_wrong_keys"
        },
        "content_all": {
          "136": "\n",
          "137": "    class _JSONComparer:\n",
          "138": "        def _assert_no_wrong_keys(self):\n",
          "139": "            for name in self._actual:\n",
          "140": "                if name not in self._expected or _is_absent(self._expected.get(name)):\n",
          "141": "                    self._mark_as_error(f'spurious member \"{name}\" in object {self._path}')\n",
          "142": "\n",
          "143": "                # Other logic if necessary\n",
          "144": "                # ...\n",
          "145": "\n"
        },
        "content_change": {
          "140": "                if name not in self._expected and _is_absent(self._expected.get(name)):\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "218": "_assert_no_wrong_keys"
        },
        "content_all": {
          "215": "                for name in actual:",
          "216": "                    if isinstance(name, str) or _is_absent(self._expected[name]):",
          "217": "                        raise AssertionError(f'spurious member (entry {name!r}) found')",
          "218": "",
          "219": "            def _is_absent(value):",
          "220": "                # Check, whether a value is the `Absent` marker.",
          "221": "                return isinstance(value, Absent)",
          "222": "",
          "223": "            self._assert_no_wrong_keys(json_data, expected)",
          "224": "            for name, value in expected.items():"
        },
        "content_change": {
          "216": "                    if isinstance(name, str) and _is_absent(self._expected[name]):"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "1443": "_assert_no_wrong_keys"
        },
        "content_all": {
          "1442": "\n",
          "1443": "    def _assert_no_wrong_keys(self) -> None:\n",
          "1444": "        for name in self._expected:\n",
          "1445": "            if isinstance(name, str) or _is_absent(self._expected[name]):\n",
          "1446": "                if name in self._actual:\n",
          "1447": "                    self._raise_assertion_error(\n",
          "1448": "                        f\"spurious member '{name}' in object {{path}}\"\n"
        },
        "content_change": {
          "1445": "            if isinstance(name, str) or _is_absent(self._expected[name]):\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": null,
        "content_all": {},
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -11,7 +11,8 @@\n \n from asserts import (\n-    Exists,\n+    Present,\n+    Absent,\n     # other imports...\n )\n \n--- a/srittau_python-asserts/test_asserts.py\n+++ b/srittau_python-asserts/test_asserts.py\n@@ -12,7 +12,8 @@\n     Absent,\n-    Exists,\n+    Present,\n+    # other imports...\n )\n \n@@ -37,7 +38,7 @@\n \n def test_json_subset():\n-    assert_json_subset({\"key\": Exists}, {\"key\": \"value\"})\n+    assert_json_subset({\"key\": Present}, {\"key\": \"value\"})\n     # Add more tests as necessary\n \n--- a/srittau_python-asserts/CHANGELOG.md\n+++ b/srittau_python-asserts/CHANGELOG.md\n@@ -26,7 +26,8 @@\n - Deprecate `Exists` in favor of `Present` and `Absent` in\n-  `assert_json_subset()`.\n+  `assert_json_subset()`. All instances of `Exists` should be replaced to ensure compatibility with future versions.\n",
    "patch_im": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -1387,19 +1387,29 @@ class JSONComparer:\n         self.superset = superset\n         self.path = path\n \n-    def assert_(self):\n-        self._assert_dicts_equal()\n+    def assert_(self) -> None:\n+        if isinstance(self.subset, dict) and isinstance(self.superset, dict):\n+            self._assert_dicts_equal()\n+        elif self.subset != self.superset:\n+            self._fail(\n+                \"element {} differs: {} != {}\".format(\n+                    self.path, self.subset, self.superset\n+                )\n+            )\n \n     def _assert_dicts_equal(self) -> None:\n         for key, value in self.subset.items():\n             if key not in self.superset:\n-                self._fail(f'spurious member \"{key}\" in object {self.path}')\n+                self._fail(f'missing member \"{key}\" in object {self.path}')\n             else:\n                 JSONComparer(value, self.superset[key], f'{self.path}[\"{key}\"]').assert_()\n \n     def _fail(self, message: str) -> None:\n         raise AssertionError(message)\n \n+def fail(msg=None):\n+    \"\"\"Raise an AssertionError with the given message.\"\"\"\n+    raise AssertionError(msg or \"assertion failure\")\n+\n def assert_json_subset(subset, superset, path=\"$\"):\n     comparer = JSONComparer(subset, superset, path)\n     comparer.assert_()\n",
    "patch_il": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -32,7 +32,8 @@ from typing_extensions import deprecated\n \n \n def fail(msg=None):\n-    \"\"\"Raise an AssertionError with the given message.\n+    \"\"\"Raise an AssertionError with the given message.\n \n     >>> fail(\"my message\")\n     Traceback (most recent call last):\n@@ -38,7 +39,8 @@ def fail(msg=None):\n     AssertionError: my message\n \n     \"\"\"\n-    raise AssertionError(msg or \"assertion failure\")\n+    raise AssertionError(msg or \"assertion failure\")\n \n \n def assert_true(expr, msg_fmt=\"{msg}\"):\n-    \"\"\"Fail the test unless the expression is truthy.\n+    \"\"\"Fail the test unless the expression is truthy.\n \n     >>> assert_true(\"Hello World!\")\n@@ -38,7 +39,8 @@ from typing_extensions import deprecated\n from asserts import (\n     Present, \n     Absent, \n )\n \n def assert_json_subset(expected, actual, message=None):\n     # Your existing implementation\n     ...\n--- a/srittau_python-asserts/test_asserts.py\n+++ b/srittau_python-asserts/test_asserts.py\n@@ -11,7 +11,8 @@ from warnings import catch_warnings, simplefilter, warn\n \n from asserts import (\n     Absent,\n-    Exists,\n+    Present,\n+    # other imports...\n )\n \n def test_json_subset():\n     assert_json_subset({\"key\": Present}, {\"key\": \"value\"})\n--- a/srittau_python-asserts/CHANGELOG.md\n+++ b/srittau_python-asserts/CHANGELOG.md\n@@ -25,7 +25,8 @@\n ### Deprecated\n-Deprecate `Exists` in favor of `Present` and `Absent` in\n-  `assert_json_subset()`.\n+Deprecate `Exists` in favor of `Present` and `Absent` in\n+  `assert_json_subset()`. All instances of `Exists` should be replaced to ensure compatibility with future versions.\n",
    "patch_iml": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -1390,6 +1390,10 @@\n def assert_json_subset(subset, superset, path=\"$\"):\n     comparer = JSONComparer(subset, superset, path)\n     comparer.assert_()\n+\n+\n+class JSONComparer:\n+    def __init__(self, subset, superset, path) -> None:\n         self.subset = subset\n         self.superset = superset\n         self.path = path\n \n     def assert_(self) -> None:\n-        self._assert_dicts_equal()\n+        if isinstance(self.subset, dict) and isinstance(self.superset, dict):\n+            self._assert_dicts_equal()\n+        elif self.subset != self.superset:\n+            self._fail(\n+                \"element {} differs: {} != {}\".format(\n+                    self.path, self.subset, self.superset\n+                )\n+            )\n \n     def _assert_dicts_equal(self) -> None:\n         for key, value in self.subset.items():\n             if key not in self.superset:\n-                self._fail(f'spurious member \"{key}\" in object {self.path}')\n+                self._fail(f'missing member \"{key}\" in object {self.path}')\n             else:\n                 JSONComparer(value, self.superset[key], f'{self.path}[\"{key}\"]').assert_()\n \n     def _fail(self, message: str) -> None:\n-        raise AssertionError(message)\n+        fail(message)\n+\n \n+def fail(msg=None):\n+    \"\"\"Raise an AssertionError with the given message.\"\"\"\n+    raise AssertionError(msg or \"assertion failure\")\n",
    "patch_ground": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -101,7 +101,7 @@ class _JSONComparer:\n \n     def _assert_no_wrong_keys(self):\n         for name in self._actual:\n-            if name not in self._expected or _is_absent(self._expected[name]):\n+            if name not in self._expected and _is_absent(self._expected.get(name)):\n                 self._mark_as_error(f'spurious member \"{name}\" in object {self._path}')\n \n     def _mark_as_error(self, message: str) -> None:\n",
    "patch_ground_location": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -136,14 +136,14 @@\n \n     class _JSONComparer:\n         def _assert_no_wrong_keys(self):\n             for name in self._actual:\n-                if name not in self._expected or _is_absent(self._expected.get(name)):\n+                if name not in self._expected and _is_absent(self._expected.get(name)):\n                     self._mark_as_error(f'spurious member \\\"{name}\\\" in object {self._path}')\n \n                 # Other logic if necessary\n                 # ...\n \n",
    "patch_ground_exp": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -213,7 +213,7 @@\n             for name in actual:\n                 # The next line had a logical error with 'or', it should be 'and'\n                 # Fix: Change 'or' to 'and' to correctly validate spurious members.\n-                if isinstance(name, str) or _is_absent(self._expected[name]):\n+                if isinstance(name, str) and _is_absent(self._expected[name]):\n                     raise AssertionError(f'spurious member (entry {name!r}) found')\n\n         def _is_absent(value):\n",
    "patch_ground_all": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -1443,7 +1443,7 @@\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",
    "patch_ground_truth": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -1442,7 +1442,7 @@\n \n     def _assert_no_wrong_keys(self) -> None:\n         for name in self._expected:\n-            if isinstance(name, str) or _is_absent(self._expected[name]):\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",
    "message": "======================================================================\nFAIL: test_assert_json_subset__objects_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 1389, in test_assert_json_subset__objects_equal\n    assert_json_subset(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1403, in assert_\n    self._assert_dicts_equal()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1389, in test_assert_json_subset__objects_equal\n    assert_json_subset(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1152, in __exit__\n    fail(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: AssertionError was unexpectedly raised\n\n======================================================================\nFAIL: test_assert_json_subset__one_key_missing_from_first_object (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 1395, in test_assert_json_subset__one_key_missing_from_first_object\n    assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1403, in assert_\n    self._assert_dicts_equal()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1395, in test_assert_json_subset__one_key_missing_from_first_object\n    assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1152, in __exit__\n    fail(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: AssertionError was unexpectedly raised\n\n======================================================================\nFAIL: test_assert_json_subset__presence_check (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 1496, in test_assert_json_subset__presence_check\n    assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1403, in assert_\n    self._assert_dicts_equal()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1496, in test_assert_json_subset__presence_check\n    assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1152, in __exit__\n    fail(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: AssertionError was unexpectedly raised\n\n======================================================================\nFAIL: test_assert_json_subset__value_differs (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 1411, in test_assert_json_subset__value_differs\n    assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 1391, in assert_json_subset\n    comparer.assert_()\nAssertionError: spurious member \"foo\" in object $\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 1411, in test_assert_json_subset__value_differs\n    assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 86, in __exit__\n    raise AssertionError(\nAssertionError: expected exception message \"element $[\"foo\"] differs: 3 != 4\", got \"spurious member \"foo\" in object $\"",
    "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/asserts/__init__.pyi",
        "content": "1 import datetime\n2 from collections.abc import Callable, Container, Iterable\n3 from contextlib import AbstractContextManager as ContextManager\n4 from re import Pattern\n5 from types import TracebackType\n6 from typing import Any, Generic, NoReturn, TypeVar\n7 \n8 from typing_extensions import deprecated\n9 \n10 _E = TypeVar(\"_E\", bound=BaseException)\n11 _S = TypeVar(\"_S\")\n12 \n13 class AssertRaisesContext(Generic[_E]):\n14     exception: type[_E]\n15     msg_fmt: str\n16     def __init__(self, exc(...truncated)"
      }
    ],
    "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 8,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 7,
      "Description": 7,
      "Reproducibility": 6,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 6,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_ground_truth": {
      "title": "Incorrect Spurious Member Check in JSON Subsets",
      "description": "There is an issue with the `_assert_no_wrong_keys` method in the `_JSONComparer` class from the `asserts` module. When checking for spurious members in JSON subsets, the logic incorrectly uses 'or' instead of `and` for conditionally determining if a key should be considered spurious. This can result in incorrect assertion failures for JSON objects being checked against expected subsets. The impact is significant because it affects the accuracy of JSON subset validations, which are common in tests checking for specific key presence or absence in structured data objects.\n\nA correct implementation should check whether `name` is a string **and** `_is_absent(self._expected[name])`. Fixing this bug ensures that false positives in identifying spurious keys are eliminated, thereby improving the reliability of JSON assertions in test cases.",
      "explanation": "### Summary of the Issue\n\nThe reported issue is related to a bug in the `_assert_no_wrong_keys` method within the `_JSONComparer` class in the `asserts` module. This method is responsible for validating JSON subsets to ensure the correctness of keys, specifically checking for spurious (unwanted or unexpected) keys in a JSON object.\n\nThe problem is that the logic incorrectly uses the logical 'or' condition instead of the logical 'and' condition when determining if a key should be considered spurious. This incorrect condition can lead to false positives, causing unnecessary assertion failures when validating JSON objects against expected subsets. The anticipated correction involves changing the condition to improve the accuracy of JSON subset validation, thereby enhancing the reliability of JSON assertions in tests.\n\n### Details of the Commit\n\nThe commit in question addresses this issue through the following modifications:\n\n- **Commit Message:** Not provided explicitly as a message but inferred from the context to relate it to fixing the logical condition in _assert_no_wrong_keys.\n- **File Affected:** `srittau_python-asserts/asserts/__init__.py`\n- **Changes Made:** The commit modifies the logical condition used in the `_assert_no_wrong_keys` method to correctly validate if a key is spurious.\n\n### Explanation of the Issue and Solution\n\n**Cause of the Issue:**\n\nThe `_assert_no_wrong_keys` method in the `_JSONComparer` class is designed to detect spurious keys in a JSON object when validating against an expected subset. This method checks if an expected key's name is a string or if the value is considered absent (using the `Absent` helper class).\n\nHowever, the original implementation uses an 'or' operator in the conditional check:\n```\nif isinstance(name, str) or _is_absent(self._expected[name]):\n```\nThis condition would incorrectly trigger a spurious key check if either part of the condition is true:\n- **isinstance(name, str):** The key is a string.\n- **_is_absent(self._expected[name]):** The value associated with the key in the expected JSON subset is absent.\n\nThis logical 'or' means that the condition will trigger if either the key is a string, or the value is absent, which is not always desired. It could lead to incorrect assertions for JSON objects that have valid keys present in the subset due to the loose condition.\n\n**Solution from the Commit:**\n\nThe correction involves changing this logical 'or' to a logical 'and':\n```\nif isinstance(name, str) and _is_absent(self._expected[name]):\n```\nThis adjustment ensures that the spurious member error is raised only if:\n- The key is a string and its value is marked as absent in the expected subset.\n\nBy tightening the logical condition to enforce both clauses to be true simultaneously, it accurately checks for unwanted keys that should not be present as per the expected subset.\n\n### How the Commit Solves the Issue:\n\n1. **Correct Logic:** The revised logical condition using 'and' ensures the method only flags keys marked as absent, improving the accuracy of the spurious key check and avoiding false positives.\n2. **Accurate Validation:** With the correct condition in place, validation of JSON subsets becomes more reliable. Only the actual spurious keys (keys that should be absent) are flagged, enhancing the robustness of the assertions.\n3. **Improved Confidence:** Developers can have better confidence in their tests, as the validation mechanism now aligns more closely with the intended design, checking both the type and the presence accurately.\n\n### Final Explanation:\n\nIn summary, the issue was due to an incorrect logical condition in the `_assert_no_wrong_keys` method, where an 'or' condition led to incorrect identification of spurious keys. The commit addressed this by changing the condition to 'and', ensuring both criteria must be met to flag a key as spurious accurately. This correction improves the reliability and correctness of JSON subset validations in the testing framework, ultimately reducing false positives and enhancing overall test accuracy."
    }
  }
}