From 0150bdd0b88a82a99e4dc957ae009579eea9517e Mon Sep 17 00:00:00 2001 From: davfsa Date: Thu, 29 Oct 2020 10:19:19 +0100 Subject: [PATCH] Enum tests - Remove unreachable and duplicate code - Fixed some logic --- coverage.ini | 7 +- hikari/internal/enums.py | 48 +++------- tests/hikari/internal/test_enums.py | 140 +++++++++++++++++++++++++++- 3 files changed, 156 insertions(+), 39 deletions(-) diff --git a/coverage.ini b/coverage.ini index 7fb6f3be9b..1f5f1727f3 100644 --- a/coverage.ini +++ b/coverage.ini @@ -7,14 +7,12 @@ source = hikari omit = hikari/__main__.py hikari/cli.py - hikari/_about.py - hikari/utilities/art.py - .nox/* [report] precision = 2 show_missing = true -skip_covered = False +skip_covered = false +sort = cover exclude_lines = \#\s*pragma: no cover ^\s*raise AssertionError\b @@ -26,4 +24,3 @@ exclude_lines = ^\s*\.\.\.$ ^\s*@abc.abstractmethod$ ^\s*if typing.TYPE_CHECKING:$ -sort = Cover diff --git a/hikari/internal/enums.py b/hikari/internal/enums.py index bc2f3f8395..df74d54ca3 100644 --- a/hikari/internal/enums.py +++ b/hikari/internal/enums.py @@ -44,13 +44,6 @@ def __init__(self, base: typing.Type[typing.Any]) -> None: self.values_to_names: typing.Dict[str, typing.Any] = {} self["__doc__"] = "An enumeration." - def __contains__(self, item: typing.Any) -> bool: - try: - _ = self[item] - return True - except KeyError: - return False - def __getitem__(self, name: str) -> typing.Any: try: return super().__getitem__(name) @@ -60,10 +53,6 @@ def __getitem__(self, name: str) -> typing.Any: except KeyError: raise KeyError(name) from None - def __iter__(self) -> typing.Iterator[str]: - yield from super().__iter__() - yield self.names_to_values - def __setitem__(self, name: str, value: typing.Any) -> None: if name == "" or name == "mro": raise TypeError(f"Invalid enum member name: {name!r}") @@ -97,8 +86,6 @@ def __setitem__(self, name: str, value: typing.Any) -> None: # We must have defined some alias, so just register the name self.names_to_values[name] = value return - if not isinstance(value, self.base): - raise TypeError("Enum values must be an instance of the base type of the enum") self.names_to_values[name] = value self.values_to_names[value] = name @@ -122,8 +109,15 @@ def __call__(cls, value: typing.Any) -> typing.Any: def __getitem__(cls, name: str) -> typing.Any: return cls._name_to_member_map_[name] + def __contains__(cls, item: typing.Any) -> bool: + try: + cls._value_to_member_map_[item] + return True + except KeyError: + return False + def __iter__(cls) -> typing.Iterator[str]: - yield from cls._name_to_member_map_ + yield from cls._name_to_member_map_.values() @staticmethod def __new__( @@ -183,7 +177,7 @@ def __prepare__( if _Enum is NotImplemented: if name != "Enum": raise TypeError("First instance of _EnumMeta must be Enum") - return {} + return _EnumNamespace(object) try: # Fails if Enum is not defined. We check this in `__new__` properly. @@ -191,17 +185,10 @@ def __prepare__( if isinstance(base, _EnumMeta): raise TypeError("First base to an enum must be the type to combine with, not _EnumMeta") - if not isinstance(enum_type, _EnumMeta): - raise TypeError("Second base to an enum must be the enum type (derived from _EnumMeta) to be used") - - if not issubclass(enum_type, _Enum): - raise TypeError("second base type for enum must be derived from Enum") return _EnumNamespace(base) except ValueError: - if name == "Enum" and _Enum is NotImplemented: - return _EnumNamespace(object) - raise TypeError("Expected two base classes for an enum") from None + raise TypeError("Expected exactly two base classes for an enum") from None def __repr__(cls) -> str: return f"" @@ -377,13 +364,10 @@ def __prepare__( raise TypeError("First instance of _FlagMeta must be Flag") return _EnumNamespace(object) - try: - # Fails if Enum is not defined. - if len(bases) == 1 and bases[0] == Flag: - return _EnumNamespace(int) - except ValueError: - pass - raise TypeError("Cannot define another Flag base type") from None + # Fails if Flag is not defined. + if len(bases) == 1 and bases[0] == Flag: + return _EnumNamespace(int) + raise TypeError("Cannot define another Flag base type") @staticmethod def __new__( @@ -761,9 +745,7 @@ def __rsub__(self: _T, other: typing.Union[int, _T]) -> _T: # This logic has to be reversed to be correct, since order matters for # a subtraction operator. This also ensures `int - _T -> _T` is a valid # case for us. - if not isinstance(other, self.__class__): - other = self.__class__(other) - return other - self + return self.__class__(other) - self def __str__(self) -> str: return self.name diff --git a/tests/hikari/internal/test_enums.py b/tests/hikari/internal/test_enums.py index 4eed26eaaf..b830054488 100644 --- a/tests/hikari/internal/test_enums.py +++ b/tests/hikari/internal/test_enums.py @@ -18,6 +18,8 @@ # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE # SOFTWARE. +import builtins + import mock import pytest @@ -74,6 +76,24 @@ def test_init_enum_type_with_bases_in_wrong_order_is_TypeError(self, args, kwarg class Enum(*args, **kwargs): pass + def test_init_with_more_than_2_types(self): + with pytest.raises(TypeError): + + class Enum(enums.Enum, str, int): + pass + + def test_init_with_less_than_2_types(self): + with pytest.raises(TypeError): + + class Enum(enums.Enum): + pass + + def test_init_with_less_than_2_types(self): + with pytest.raises(TypeError): + + class Enum(enums.Enum, metaclass=enums._EnumMeta): + pass + def test_init_enum_type_default_docstring_set(self): class Enum(str, enums.Enum): pass @@ -160,6 +180,26 @@ def p(self): assert Enum.__members__ == {"foo": 9, "bar": 18, "baz": 27} + def test_init_with_invalid_name(self): + with pytest.raises(TypeError): + + class Enum(int, enums.Enum): + mro = 420 + + def test_init_with_unhashable_value(self): + with mock.patch.object(builtins, "hash", side_effect=TypeError): + with pytest.raises(TypeError): + + class Enum(int, enums.Enum): + test = 420 + + def test_init_with_duplicate(self): + with pytest.raises(TypeError): + + class Enum(int, enums.Enum): + test = 123 + test = 321 + def test_call_when_member(self): class Enum(int, enums.Enum): foo = 9 @@ -190,6 +230,44 @@ class Enum(int, enums.Enum): assert returned == Enum.foo assert type(returned) == Enum + def test_contains(self): + class Enum(int, enums.Enum): + foo = 9 + bar = 18 + baz = 27 + + assert 9 in Enum + assert 100 not in Enum + + def test_name(self): + class Enum(int, enums.Enum): + foo = 9 + bar = 18 + baz = 27 + + assert Enum.foo.name == "foo" + + def test_iter(self): + class Enum(int, enums.Enum): + foo = 9 + bar = 18 + baz = 27 + + a = [] + for i in Enum: + a.append(i) + + assert a == [Enum.foo, Enum.bar, Enum.baz] + + def test_repr(self): + class Enum(int, enums.Enum): + foo = 9 + bar = 18 + baz = 27 + + assert repr(Enum) == "" + assert repr(Enum.foo) == "" + class TestIntFlag: @mock.patch.object(enums, "_Flag", new=NotImplemented) @@ -260,7 +338,7 @@ class Flag(enums.Flag): def foo(self): return "foo" - assert Flag.foo(12) == "foo" + assert Flag().foo() == "foo" def test_init_flag_type_allows_classmethods(self): class Flag(enums.Flag): @@ -461,6 +539,29 @@ class Flag(enums.Flag): assert Flag._temp_members_ == {3: Flag.foo | Flag.bar, 7: Flag.foo | Flag.bar | Flag.baz} assert Flag._temp_members_ == {3: Flag.foo | Flag.bar, 7: Flag.foo | Flag.bar | Flag.baz} + def test_cache_when_temp_values_over_MAX_CACHED_MEMBERS(self): + class MockDict: + def __getitem__(self, key): + raise KeyError + + def __len__(self): + return enums._MAX_CACHED_MEMBERS + 1 + + def __setitem__(self, k, v): + pass + + popitem = mock.Mock() + + class Flag(enums.Flag): + foo = 1 + bar = 2 + baz = 3 + + Flag._temp_members_ = MockDict() + + Flag(4) + Flag._temp_members_.popitem.assert_called_once_with() + def test_bitwise_name(self): class Flag(enums.Flag): foo = 1 @@ -1088,3 +1189,40 @@ class TestFlag(enums.Flag): assert isinstance(0x1C ^ a, TestFlag) assert 0x1C ^ a == TestFlag.FOO | TestFlag.BAR | TestFlag.BORK | TestFlag.QUX assert 0x1C ^ a == 0x1B + + def test_iter(self): + class TestFlag(enums.Flag): + FOO = 0x1 + BAR = 0x2 + BAZ = 0x4 + BORK = 0x8 + QUX = 0x10 + + a = [] + for i in TestFlag: + a.append(i) + + assert a == [TestFlag.FOO, TestFlag.BAR, TestFlag.BAZ, TestFlag.BORK, TestFlag.QUX] + + def test_getitem(self): + class TestFlag(enums.Flag): + FOO = 0x1 + BAR = 0x2 + BAZ = 0x4 + BORK = 0x8 + QUX = 0x10 + + returned = TestFlag["FOO"] + assert returned == TestFlag.FOO + assert type(returned) == TestFlag + + def test_repr(self): + class TestFlag(enums.Flag): + FOO = 0x1 + BAR = 0x2 + BAZ = 0x4 + BORK = 0x8 + QUX = 0x10 + + assert repr(TestFlag) == "" + assert repr(TestFlag.FOO) == ""