mirror of https://github.com/jab/bidict.git
150 lines
6.0 KiB
Python
150 lines
6.0 KiB
Python
# Copyright 2009-2022 Joshua Bronson. All rights reserved.
|
|
#
|
|
# This Source Code Form is subject to the terms of the Mozilla Public
|
|
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
"""Test various issubclass checks."""
|
|
|
|
import sys
|
|
from collections.abc import Hashable, Mapping, MutableMapping, Reversible
|
|
from collections import OrderedDict
|
|
|
|
import pytest
|
|
|
|
from bidict import (
|
|
bidict, frozenbidict, namedbidict, FrozenOrderedBidict, OrderedBidict,
|
|
BidirectionalMapping, MutableBidirectionalMapping,
|
|
BidictBase, MutableBidict, OrderedBidictBase,
|
|
NamedBidictBase, GeneratedBidictInverse,
|
|
)
|
|
|
|
|
|
class AbstractBimap(BidirectionalMapping):
|
|
"""Does not override `inverse` and therefore should not be instantiatable."""
|
|
|
|
|
|
BIDICT_BASE_TYPES = (BidictBase, MutableBidict, OrderedBidictBase)
|
|
BIDICT_TYPES = BIDICT_BASE_TYPES + (bidict, frozenbidict, FrozenOrderedBidict, OrderedBidict)
|
|
MyNamedBidict = namedbidict('MyNamedBidict', 'key', 'val')
|
|
BIMAP_TYPES = BIDICT_TYPES + (AbstractBimap, MyNamedBidict)
|
|
NOT_BIMAP_TYPES = (dict, OrderedDict, int, object)
|
|
MUTABLE_BIDICT_TYPES = (bidict, OrderedBidict)
|
|
HASHABLE_BIDICT_TYPES = (frozenbidict, FrozenOrderedBidict)
|
|
ORDERED_BIDICT_TYPES = (OrderedBidict, FrozenOrderedBidict)
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', BIMAP_TYPES)
|
|
def test_issubclass_bimap(bi_cls):
|
|
"""All bidict types should be considered subclasses of :class:`BidirectionalMapping`."""
|
|
assert issubclass(bi_cls, BidirectionalMapping)
|
|
|
|
|
|
@pytest.mark.parametrize('not_bi_cls', NOT_BIMAP_TYPES)
|
|
def test_not_issubclass_not_bimap(not_bi_cls):
|
|
"""Classes that do not conform to :class:`BidirectionalMapping` interface
|
|
should not be considered subclasses of it.
|
|
"""
|
|
assert not issubclass(not_bi_cls, BidirectionalMapping)
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', BIDICT_TYPES)
|
|
def test_issubclass_mapping(bi_cls):
|
|
"""All bidict types should be :class:`collections.abc.Mapping`s."""
|
|
assert issubclass(bi_cls, Mapping)
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', MUTABLE_BIDICT_TYPES)
|
|
def test_issubclass_mutable_and_mutable_bidirectional_mapping(bi_cls):
|
|
"""All mutable bidict types should be mutable (bidirectional) mappings."""
|
|
assert issubclass(bi_cls, MutableMapping)
|
|
assert issubclass(bi_cls, MutableBidirectionalMapping)
|
|
|
|
|
|
def test_issubclass_namedbidict():
|
|
"""Named bidicts should derive from NamedBidictBase and their inverse classes from GeneratedBidictInverse."""
|
|
assert issubclass(MyNamedBidict, NamedBidictBase)
|
|
assert issubclass(MyNamedBidict._inv_cls, GeneratedBidictInverse)
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', HASHABLE_BIDICT_TYPES)
|
|
def test_hashable_not_mutable(bi_cls):
|
|
"""All hashable bidict types should not be mutable (bidirectional) mappings."""
|
|
assert not issubclass(bi_cls, MutableMapping)
|
|
assert not issubclass(bi_cls, MutableBidirectionalMapping)
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', HASHABLE_BIDICT_TYPES)
|
|
def test_issubclass_hashable(bi_cls):
|
|
"""All hashable bidict types should implement :class:`collections.abc.Hashable`."""
|
|
assert issubclass(bi_cls, Hashable)
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', ORDERED_BIDICT_TYPES)
|
|
def test_ordered_reversible(bi_cls):
|
|
"""All ordered bidict types should be reversible."""
|
|
assert issubclass(bi_cls, Reversible)
|
|
|
|
|
|
def test_issubclass_internal():
|
|
"""The docs specifically recommend using ABCs
|
|
over concrete classes when checking whether an interface is provided
|
|
(see :ref:`polymorphism`).
|
|
|
|
The relationships tested here are not guaranteed to hold in the future,
|
|
but are still tested so that any unintentional changes won't go unnoticed.
|
|
"""
|
|
assert not issubclass(bidict, FrozenOrderedBidict)
|
|
assert not issubclass(bidict, OrderedBidict)
|
|
assert not issubclass(bidict, frozenbidict)
|
|
|
|
assert not issubclass(FrozenOrderedBidict, OrderedBidict)
|
|
assert not issubclass(FrozenOrderedBidict, bidict)
|
|
assert not issubclass(FrozenOrderedBidict, frozenbidict)
|
|
|
|
assert not issubclass(OrderedBidict, FrozenOrderedBidict)
|
|
assert not issubclass(OrderedBidict, bidict)
|
|
assert not issubclass(OrderedBidict, frozenbidict)
|
|
|
|
assert not issubclass(frozenbidict, FrozenOrderedBidict)
|
|
assert not issubclass(frozenbidict, OrderedBidict)
|
|
assert not issubclass(frozenbidict, bidict)
|
|
|
|
# Regression test for #111, Bug in BidirectionalMapping.__subclasshook__():
|
|
# Any class with an inverse attribute is considered a collections.abc.Mapping
|
|
OnlyHasInverse = type('OnlyHasInverse', (), {'inverse': 'foo'})
|
|
assert not issubclass(OnlyHasInverse, Mapping)
|
|
|
|
|
|
def test_abstract_bimap_init_fails():
|
|
"""Instantiating `AbstractBimap` should fail with expected TypeError."""
|
|
excmatch = r"Can't instantiate abstract class AbstractBimap with abstract methods .* inverse"
|
|
with pytest.raises(TypeError, match=excmatch):
|
|
AbstractBimap() # pylint: disable=abstract-class-instantiated
|
|
|
|
|
|
def test_bimap_inverse_notimplemented():
|
|
"""Calling .inverse on a BidirectionalMapping should raise :class:`NotImplementedError`."""
|
|
with pytest.raises(NotImplementedError):
|
|
# Can't instantiate a BidirectionalMapping that hasn't overridden the abstract methods of
|
|
# the interface, so only way to call this implementation is on the class.
|
|
BidirectionalMapping.inverse.fget(bidict())
|
|
|
|
|
|
@pytest.mark.parametrize('bi_cls', BIDICT_BASE_TYPES)
|
|
def test_bidict_bases_init_succeed(bi_cls):
|
|
"""Bidict base classes should be initializable and have a working .inverse property."""
|
|
b = bi_cls(one=1, two=2)
|
|
assert dict(b.inverse) == {1: 'one', 2: 'two'}
|
|
|
|
|
|
def test_bidict_reversible_matches_dict_reversible():
|
|
"""Reversibility of bidict matches dict's on all supported Python versions."""
|
|
assert issubclass(bidict, Reversible) == issubclass(dict, Reversible)
|
|
|
|
|
|
@pytest.mark.skipif(sys.version_info < (3, 8), reason='reversible unordered bidicts require Python 3.8+')
|
|
def test_bidict_reversible():
|
|
"""All bidicts are Reversible on Python 3.8+."""
|
|
assert issubclass(bidict, Reversible)
|