bidict/tests/test_hypothesis.py

414 lines
17 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2018-01-01 03:51:19 +00:00
# Copyright 2018 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/.
"""Property-based tests using https://hypothesis.readthedocs.io."""
2017-12-18 02:42:46 +00:00
import gc
2018-02-19 07:53:03 +00:00
import pickle
import re
from collections import OrderedDict
from operator import eq, ne, itemgetter
from os import getenv
2017-12-18 02:42:46 +00:00
from weakref import ref
2016-06-28 04:05:22 +00:00
import pytest
from hypothesis import assume, given, settings, strategies as strat
from bidict import (
BidictException, IGNORE, OVERWRITE, RAISE,
BidirectionalMapping, bidict, OrderedBidict, OrderedBidictBase,
frozenbidict, FrozenOrderedBidict, namedbidict, pairs, inverted)
from bidict.compat import (
PY2, PYPY, iterkeys, itervalues, iteritems, izip,
Hashable, Mapping, MutableMapping)
settings.register_profile('default', settings(max_examples=250, deadline=None))
settings.load_profile(getenv('HYPOTHESIS_PROFILE', 'default'))
def inverse_odict(items):
"""An OrderedDict containing the inverse of each item in *items*."""
2016-06-28 04:05:22 +00:00
return OrderedDict((v, k) for (k, v) in items)
def ensure_no_dup(items):
"""Given some hypothesis-generated items, prune any with duplicated keys or values."""
pruned = list(iteritems(inverse_odict(iteritems(inverse_odict(items)))))
assume(len(pruned) >= len(items) // 2)
return pruned
def ensure_dup(key=False, val=False):
"""Return a function that takes some hypothesis-generated items
and ensures they contain the specified type of duplication.
"""
assert key or val
def _wrapped(items): # noqa: E306 (expected 1 blank line before a nested definition)
fwd = dict(items)
if key:
assume(len(fwd) < len(items))
if val:
inv = dict((v, k) for (k, v) in items)
assume(len(inv) < len(items))
if key and val:
invinv = dict((v, k) for (k, v) in iteritems(inv))
# If an item has a duplicate key and val, they must duplicate two other distinct items.
assume(len(invinv) < len(fwd))
return items
return _wrapped
KEY = itemgetter(0)
MyNamedBidict = namedbidict('MyNamedBidict', 'key', 'val')
MyNamedFrozenBidict = namedbidict('MyNamedBidict', 'key', 'val', base_type=frozenbidict)
NAMEDBIDICT_VALID_NAME = re.compile('^[A-z][A-z0-9_]*$')
MUTABLE_BIDICT_TYPES = (
bidict, OrderedBidict, MyNamedBidict)
IMMUTABLE_BIDICT_TYPES = (frozenbidict, FrozenOrderedBidict, MyNamedFrozenBidict)
ORDERED_BIDICT_TYPES = (OrderedBidict, FrozenOrderedBidict)
BIDICT_TYPES = MUTABLE_BIDICT_TYPES + IMMUTABLE_BIDICT_TYPES
MAPPING_TYPES = BIDICT_TYPES + (dict, OrderedDict)
H_BIDICT_TYPES = strat.sampled_from(BIDICT_TYPES)
H_MUTABLE_BIDICT_TYPES = strat.sampled_from(MUTABLE_BIDICT_TYPES)
H_IMMUTABLE_BIDICT_TYPES = strat.sampled_from(IMMUTABLE_BIDICT_TYPES)
H_ORDERED_BIDICT_TYPES = strat.sampled_from(ORDERED_BIDICT_TYPES)
H_MAPPING_TYPES = strat.sampled_from(MAPPING_TYPES)
H_NAMES = strat.sampled_from(('valid1', 'valid2', 'valid3', 'in-valid'))
H_DUP_POLICIES = strat.sampled_from((IGNORE, OVERWRITE, RAISE))
H_BOOLEANS = strat.booleans()
H_TEXT = strat.text()
H_NONE = strat.none()
H_IMMUTABLES = H_BOOLEANS | H_TEXT | H_NONE | strat.integers() | strat.floats(allow_nan=False)
H_NON_MAPPINGS = H_NONE
H_PAIRS = strat.tuples(H_IMMUTABLES, H_IMMUTABLES)
H_LISTS_PAIRS = strat.lists(H_PAIRS)
H_LISTS_PAIRS_NODUP = H_LISTS_PAIRS.map(ensure_no_dup)
H_LISTS_PAIRS_DUP = (
H_LISTS_PAIRS.map(ensure_dup(key=True)) |
H_LISTS_PAIRS.map(ensure_dup(val=True)) |
H_LISTS_PAIRS.map(ensure_dup(key=True, val=True)))
H_TEXT_PAIRS = strat.tuples(H_TEXT, H_TEXT)
H_LISTS_TEXT_PAIRS_NODUP = strat.lists(H_TEXT_PAIRS).map(ensure_no_dup)
H_METHOD_ARGS = strat.sampled_from((
# 0-arity
('clear', ()),
('popitem', ()),
# 1-arity
('__delitem__', (H_IMMUTABLES,)),
('pop', (H_IMMUTABLES,)),
('setdefault', (H_IMMUTABLES,)),
('move_to_end', (H_IMMUTABLES,)),
('update', (H_LISTS_PAIRS,)),
('forceupdate', (H_LISTS_PAIRS,)),
# 2-arity
('pop', (H_IMMUTABLES, H_IMMUTABLES)),
('setdefault', (H_IMMUTABLES, H_IMMUTABLES)),
('move_to_end', (H_IMMUTABLES, H_BOOLEANS)),
('__setitem__', (H_IMMUTABLES, H_IMMUTABLES)),
('put', (H_IMMUTABLES, H_IMMUTABLES)),
('forceput', (H_IMMUTABLES, H_IMMUTABLES)),
))
2018-02-24 05:35:44 +00:00
def items_match(map1, map2, relation=eq):
"""Return whether the items of *map1* and *map2*
are related by *relation*.
If *map1* and *map2* are both ordered,
they will be compared as lists,
otherwise as sets.
"""
both_ordered_bidicts = all(isinstance(m, OrderedBidictBase) for m in (map1, map2))
canon = list if both_ordered_bidicts else set
canon_map1 = canon(iteritems(map1))
canon_map2 = canon(iteritems(map2))
return relation(canon_map1, canon_map2)
@given(bi_cls=H_BIDICT_TYPES, other_cls=H_MAPPING_TYPES, not_a_mapping=H_NON_MAPPINGS,
init_items=H_LISTS_PAIRS_NODUP, init_unequal=H_LISTS_PAIRS_NODUP)
def test_eq_ne_hash(bi_cls, other_cls, init_items, init_unequal, not_a_mapping):
"""Test various equality comparisons and hashes between bidicts and other objects."""
assume(init_items != init_unequal)
some_bidict = bi_cls(init_items)
other_equal = other_cls(init_items)
other_equal_inv = inverse_odict(iteritems(other_equal))
assert items_match(some_bidict, other_equal)
assert items_match(some_bidict.inv, other_equal_inv)
assert some_bidict == other_equal
assert not some_bidict != other_equal
assert some_bidict.inv == other_equal_inv
assert not some_bidict.inv != other_equal_inv
has_eq_order_sens = getattr(bi_cls, 'equals_order_sensitive', None)
other_is_ordered = getattr(other_cls, '__reversed__', None)
if has_eq_order_sens and other_is_ordered:
assert some_bidict.equals_order_sensitive(other_equal)
assert some_bidict.inv.equals_order_sensitive(other_equal_inv)
both_hashable = all(issubclass(cls, Hashable) for cls in (bi_cls, other_cls))
if both_hashable:
assert hash(some_bidict) == hash(other_equal)
other_unequal = other_cls(init_unequal)
other_unequal_inv = inverse_odict(iteritems(other_unequal))
assert items_match(some_bidict, other_unequal, relation=ne)
assert items_match(some_bidict.inv, other_unequal_inv, relation=ne)
assert some_bidict != other_unequal
assert not some_bidict == other_unequal
assert some_bidict.inv != other_unequal_inv
assert not some_bidict.inv == other_unequal_inv
if has_eq_order_sens:
assert not some_bidict.equals_order_sensitive(other_unequal)
assert not some_bidict.inv.equals_order_sensitive(other_unequal_inv)
assert not some_bidict == not_a_mapping
assert not some_bidict.inv == not_a_mapping
assert some_bidict != not_a_mapping
assert some_bidict.inv != not_a_mapping
if has_eq_order_sens:
assert not some_bidict.equals_order_sensitive(not_a_mapping)
assert not some_bidict.inv.equals_order_sensitive(not_a_mapping)
@given(bi_cls=H_BIDICT_TYPES, init_items=H_LISTS_PAIRS_NODUP)
def test_bijectivity(bi_cls, init_items):
"""b[k] == v <==> b.inv[v] == k"""
some_bidict = bi_cls(init_items)
ordered = getattr(bi_cls, '__reversed__', None)
canon = list if ordered else set
keys = canon(iterkeys(some_bidict))
vals = canon(itervalues(some_bidict))
fwd_by_keys = canon(some_bidict[k] for k in iterkeys(some_bidict))
inv_by_vals = canon(some_bidict.inv[v] for v in itervalues(some_bidict))
assert keys == inv_by_vals
assert vals == fwd_by_keys
inv = some_bidict.inv
inv_keys = canon(iterkeys(inv))
inv_vals = canon(itervalues(inv))
inv_fwd_by_keys = canon(inv[k] for k in iterkeys(inv))
inv_inv_by_vals = canon(inv.inv[v] for v in itervalues(inv))
assert inv_keys == inv_inv_by_vals
assert inv_vals == inv_fwd_by_keys
@given(bi_cls=H_MUTABLE_BIDICT_TYPES, init_items=H_LISTS_PAIRS_NODUP,
method_args=H_METHOD_ARGS, data=strat.data())
def test_consistency_after_mutation(bi_cls, init_items, method_args, data):
"""Every bidict should be left in a consistent state after calling
any mutating method on it that it implements, even if the call raises.
"""
methodname, hs_args = method_args
method = getattr(bi_cls, methodname, None)
2016-06-28 04:05:22 +00:00
if not method:
return
args = tuple(data.draw(i) for i in hs_args)
bi_init = bi_cls(init_items)
bi_clone = bi_init.copy()
assert items_match(bi_init, bi_clone)
2016-06-28 04:05:22 +00:00
try:
method(bi_clone, *args)
except (KeyError, BidictException) as exc:
# Call should fail clean, i.e. bi_clone should be in the same state it was before the call.
assertmsg = '%r did not fail clean: %r' % (method, exc)
assert items_match(bi_clone, bi_init), assertmsg
assert items_match(bi_clone.inv, bi_init.inv), assertmsg
# Whether the call failed or succeeded, bi_clone should pass consistency checks.
assert len(bi_clone) == sum(1 for _ in iteritems(bi_clone))
assert len(bi_clone) == sum(1 for _ in iteritems(bi_clone.inv))
assert items_match(bi_clone, dict(bi_clone))
assert items_match(bi_clone.inv, dict(bi_clone.inv))
assert items_match(bi_clone, inverse_odict(iteritems(bi_clone.inv)))
assert items_match(bi_clone.inv, inverse_odict(iteritems(bi_clone)))
@given(bi_cls=H_MUTABLE_BIDICT_TYPES,
init_items=H_LISTS_PAIRS_NODUP,
update_items=H_LISTS_PAIRS_DUP,
on_dup_key=H_DUP_POLICIES, on_dup_val=H_DUP_POLICIES, on_dup_kv=H_DUP_POLICIES)
def test_dup_policies_bulk(bi_cls, init_items, update_items, on_dup_key, on_dup_val, on_dup_kv):
"""Attempting a bulk update with *update_items* should yield the same result as
attempting to set each of the items sequentially
while respecting the duplication policies that are in effect.
"""
dup_policies = dict(on_dup_key=on_dup_key, on_dup_val=on_dup_val, on_dup_kv=on_dup_kv)
bi_init = bi_cls(init_items)
expect = bi_init.copy()
2016-06-28 04:05:22 +00:00
expectexc = None
for (key, val) in update_items:
2016-06-28 04:05:22 +00:00
try:
expect.put(key, val, **dup_policies)
except BidictException as exc:
expectexc = type(exc)
expect = bi_init # bulk updates fail clean
2016-06-28 04:05:22 +00:00
break
check = bi_init.copy()
2016-06-28 04:05:22 +00:00
checkexc = None
try:
check.putall(update_items, **dup_policies)
except BidictException as exc:
checkexc = type(exc)
assert checkexc == expectexc
assert items_match(check, expect)
assert items_match(check.inv, expect.inv)
@given(bi_cls=H_BIDICT_TYPES, init_items=H_LISTS_PAIRS_NODUP)
def test_bidict_iter(bi_cls, init_items):
""":meth:`bidict.BidictBase.__iter__` should yield all the keys in a bidict."""
some_bidict = bi_cls(init_items)
assert set(some_bidict) == set(iterkeys(some_bidict)) == set(KEY(pair) for pair in init_items)
@given(bi_cls=H_ORDERED_BIDICT_TYPES, init_items=H_LISTS_PAIRS_NODUP)
def test_orderedbidict_iter(bi_cls, init_items):
""":meth:`bidict.OrderedBidictBase.__iter__` should yield all the keys
in an ordered bidict in the order they were inserted.
"""
some_bidict = bi_cls(init_items)
key_iters = (some_bidict, iterkeys(some_bidict), (KEY(pair) for pair in init_items))
assert all(i == j == k for (i, j, k) in izip(*key_iters))
@given(bi_cls=H_ORDERED_BIDICT_TYPES, init_items=H_LISTS_PAIRS_NODUP)
def test_orderedbidict_reversed(bi_cls, init_items):
""":meth:`bidict.OrderedBidictBase.__reversed__` should yield all the keys
in an ordered bidict in the reverse-order they were inserted.
"""
some_bidict = bi_cls(init_items)
key_iters = (some_bidict, iterkeys(some_bidict), (KEY(pair) for pair in init_items))
key_iters_rev = (reversed(list(i)) for i in key_iters)
assert all(i == j == k for (i, j, k) in izip(*key_iters_rev))
@given(bi_cls=H_IMMUTABLE_BIDICT_TYPES)
def test_frozenbidicts_hashable(bi_cls):
"""Immutable bidicts can be hashed and inserted into sets and mappings."""
some_bidict = bi_cls()
# Nothing to assert; making sure these calls don't raise TypeError is sufficient.
hash(some_bidict) # pylint: disable=pointless-statement
{some_bidict} # pylint: disable=pointless-statement
{some_bidict: some_bidict} # pylint: disable=pointless-statement
@given(base_type=H_MAPPING_TYPES, init_items=H_LISTS_PAIRS_NODUP, data=strat.data())
def test_namedbidict(base_type, init_items, data):
"""Test the :func:`bidict.namedbidict` factory and custom accessors."""
names = typename, keyname, valname = [data.draw(H_NAMES) for _ in range(3)]
try:
nbcls = namedbidict(typename, keyname, valname, base_type=base_type)
except ValueError:
# Either one of the names was invalid, or the keyname and valname were not distinct.
assert not all(map(NAMEDBIDICT_VALID_NAME.match, names)) or keyname == valname
return
except TypeError:
# The base type must not have been a BidirectionalMapping.
assert not issubclass(base_type, BidirectionalMapping)
return
assume(init_items)
instance = nbcls(init_items)
valfor = getattr(instance, valname + '_for')
keyfor = getattr(instance, keyname + '_for')
assert all(valfor[key] == val for (key, val) in iteritems(instance))
assert all(keyfor[val] == key for (key, val) in iteritems(instance))
# The same custom accessors should work on the inverse.
inv = instance.inv
valfor = getattr(inv, valname + '_for')
keyfor = getattr(inv, keyname + '_for')
assert all(valfor[key] == val for (key, val) in iteritems(instance))
assert all(keyfor[val] == key for (key, val) in iteritems(instance))
@given(bi_cls=H_BIDICT_TYPES)
def test_bidict_isinv(bi_cls):
"""All bidict types should provide ``_isinv`` and ``__getstate__``
(or else they won't fully work as a *base_type* for :func:`namedbidict`).
"""
some_bidict = bi_cls()
# Nothing to assert; making sure these calls don't raise is sufficient.
some_bidict._isinv # pylint: disable=pointless-statement,protected-access
some_bidict.__getstate__() # pylint: disable=pointless-statement
# Skip this test on PyPy where reference counting isn't used to free objects immediately. See:
# http://doc.pypy.org/en/latest/cpython_differences.html#differences-related-to-garbage-collection-strategies
2017-12-18 02:42:46 +00:00
# "It also means that weak references may stay alive for a bit longer than expected."
@pytest.mark.skipif(PYPY, reason='objects with 0 refcount not freed immediately on PyPy')
@given(bi_cls=H_BIDICT_TYPES)
def test_no_reference_cycles(bi_cls):
"""When you delete your last strong reference to a bidict,
there are no remaining strong references to it
(e.g. no reference cycle was created between it and its inverse)
so its memory can be reclaimed immediately.
"""
2017-12-18 02:42:46 +00:00
gc.disable()
some_bidict = bi_cls()
weak = ref(some_bidict)
2017-12-18 02:42:46 +00:00
assert weak() is not None
del some_bidict
2017-12-18 02:42:46 +00:00
assert weak() is None
gc.enable()
@given(bi_cls=H_BIDICT_TYPES)
def test_slots(bi_cls):
"""See https://docs.python.org/3/reference/datamodel.html#notes-on-using-slots."""
stop_at = {object}
if PY2:
stop_at.update({Mapping, MutableMapping}) # These don't define __slots__ in Python 2.
cls_by_slot = {}
for cls in bi_cls.__mro__:
if cls in stop_at:
break
slots = cls.__dict__.get('__slots__')
assert slots is not None, 'Expected %r to define __slots__' % cls
for slot in slots:
seen_at = cls_by_slot.get(slot)
assert not seen_at, '%r repeats slot %r declared first by %r' % (seen_at, slot, cls)
cls_by_slot[slot] = cls
@given(bi_cls=H_BIDICT_TYPES, init_items=H_LISTS_PAIRS_NODUP)
def test_pickle_roundtrips(bi_cls, init_items):
"""A bidict should equal the result of unpickling its pickle."""
some_bidict = bi_cls(init_items)
dumps_args = {}
# Pickling ordered bidicts in Python 2 requires a higher (non-default) protocol version.
if PY2 and issubclass(bi_cls, OrderedBidictBase):
dumps_args['protocol'] = 2
pickled = pickle.dumps(some_bidict, **dumps_args)
roundtripped = pickle.loads(pickled)
assert roundtripped == some_bidict
@given(items=H_LISTS_PAIRS, kwitems=H_LISTS_TEXT_PAIRS_NODUP)
def test_pairs(items, kwitems):
"""Test that :func:`bidict.pairs` works correctly."""
assert list(pairs(items)) == list(items)
assert list(pairs(OrderedDict(kwitems))) == list(kwitems)
kwdict = dict(kwitems)
pairs_it = pairs(items, **kwdict)
assert all(i == j for (i, j) in izip(items, pairs_it))
assert set(iteritems(kwdict)) == {i for i in pairs_it}
with pytest.raises(TypeError):
pairs('too', 'many', 'args')
@given(bi_cls=H_BIDICT_TYPES, items=H_LISTS_PAIRS_NODUP)
def test_inverted(bi_cls, items):
"""Test that :func:`bidict.inverted` works correctly."""
inv_items = [(v, k) for (k, v) in items]
assert list(inverted(items)) == inv_items
assert list(inverted(inverted(items))) == items
some_bidict = bi_cls(items)
inv_bidict = bi_cls(inv_items)
assert some_bidict.inv == inv_bidict
assert set(inverted(some_bidict)) == set(inv_items)
assert bi_cls(inverted(inv_bidict)) == some_bidict