bidict/tests/test_class_relationships.py

150 lines
6.0 KiB
Python
Raw Normal View History

2022-02-15 20:36:58 +00:00
# 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."""
2021-09-05 15:25:22 +00:00
import sys
from collections.abc import Hashable, Mapping, MutableMapping, Reversible
from collections import OrderedDict
2018-04-28 01:50:29 +00:00
import pytest
2020-08-01 13:44:09 +00:00
from bidict import (
2022-02-07 16:42:09 +00:00
bidict, frozenbidict, namedbidict, FrozenOrderedBidict, OrderedBidict,
BidirectionalMapping, MutableBidirectionalMapping,
BidictBase, MutableBidict, OrderedBidictBase,
NamedBidictBase, GeneratedBidictInverse,
2022-02-07 16:42:09 +00:00
)
2022-02-07 16:42:09 +00:00
class AbstractBimap(BidirectionalMapping):
"""Does not override `inverse` and therefore should not be instantiatable."""
BIDICT_BASE_TYPES = (BidictBase, MutableBidict, OrderedBidictBase)
2022-02-07 16:42:09 +00:00
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)
2020-07-26 17:21:27 +00:00
def test_issubclass_mutable_and_mutable_bidirectional_mapping(bi_cls):
"""All mutable bidict types should be mutable (bidirectional) mappings."""
assert issubclass(bi_cls, MutableMapping)
2020-07-26 17:21:27 +00:00
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)
2020-07-26 17:21:27 +00:00
@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."""
2022-02-07 16:42:09 +00:00
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)
Various improvements. - Refactor proxied- (i.e. delegated-) to-``_fwdm`` logic for better composability and interoperability. Drop the ``_Proxied*`` mixin classes and instead move their methods into :class:`~bidict.BidictBase`, which now checks for an object defined by the :attr:`~bidict.BidictBase.__delegate__` attribute. The :attr:`~bidict.BidictBase.__delegate__` object will be delegated to if the method is available on it, otherwise a default implementation (e.g. inherited from :class:`~collections.abc.Mapping`) will be used otherwise. Subclasses may set ``__delegate__ = None`` to opt out. - Consolidate ``_MutableBidict`` into :class:`bidict.bidict` now that the dropped mixin classes make it unnecessary. - Change :attr:`~bidict.BidictBase.__repr_delegate__` to take simply a type like :class:`dict` or :class:`list`. - Upgrade to latest major `sortedcontainers <https://github.com/grantjenks/python-sortedcontainers>`__ version (from v1 to v2) for the :ref:`extending:Sorted Bidict Recipes`. - ``bidict.compat.{view,iter}{keys,values,items}`` on Python2 no longer assumes the target object implements these methods, as they're not actually part of the :class:`~collections.abc.Mapping` interface, and provides fallback implementations when the methods are unavailable. This allows the :ref:`extending:Sorted Bidict Recipes` to continue to work with sortedcontainers v2 on Python2. - Test code in docs via Sphinx rather than pytest. Enables running Python version-dependent tests conditionally rather than skipping them altogether, as well as hiding import statements (via `testsetup`) that otherwise just add noise. Run tests (viz. pytest and Sphinx doctest) via a new run-tests.py script.
2018-11-05 20:52:59 +00:00
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
2022-03-27 20:36:17 +00:00
OnlyHasInverse = type('OnlyHasInverse', (), {'inverse': 'foo'})
assert not issubclass(OnlyHasInverse, Mapping)
2022-02-07 16:42:09 +00:00
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.
2020-08-01 13:44:09 +00:00
BidirectionalMapping.inverse.fget(bidict())
2021-09-05 15:25:22 +00:00
2022-02-07 16:42:09 +00:00
@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'}
2021-09-05 15:25:22 +00:00
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+')
2021-09-05 15:25:22 +00:00
def test_bidict_reversible():
"""All bidicts are Reversible on Python 3.8+."""
assert issubclass(bidict, Reversible)