import re
from abc import ABC
from datetime import date, datetime
from decimal import Decimal
from functools import partial
from collections import namedtuple
from testfixtures.shouldraise import ShouldAssert
from testfixtures.tests.sample1 import SampleClassA, SampleClassB, Slotted
from testfixtures.mock import Mock, call
from re import compile
from testfixtures import (
Comparison as C,
Replacer,
ShouldRaise,
compare,
generator,
singleton,
)
from testfixtures.compat import PY_37_PLUS
from testfixtures.comparison import compare_sequence, compare_object
from unittest import TestCase
hexaddr = compile('0x[0-9A-Fa-f]+')
def hexsub(raw):
return hexaddr.sub('...', raw)
call_list_repr = repr(Mock().mock_calls.__class__)
marker = object()
_compare = compare
class Lazy:
def __init__(self, message):
self.message = message
def __str__(self):
return self.message
def check_raises(x_=marker, y_=marker, message=None, regex=None,
compare=compare, **kw):
args = []
for value in x_, y_:
if value is not marker:
args.append(value)
try:
compare(*args, **kw)
except Exception as e:
if not isinstance(e, AssertionError): # pragma: no cover
raise
actual = hexsub(e.args[0])
if message is not None:
# handy for debugging, but can't be relied on for tests!
_compare(actual, expected=message, show_whitespace=True)
assert actual == message
else:
if not regex.match(actual): # pragma: no cover
raise AssertionError(
'%r did not match %r' % (actual, regex.pattern)
)
else:
raise AssertionError('No exception raised!')
class CompareHelper(object):
def check_raises(self, *args, **kw):
check_raises(*args, **kw)
class TestCompare(CompareHelper, TestCase):
def test_object_same(self):
o = object()
compare(o, o)
def test_object_diff(self):
self.check_raises(
object(), object(),
'<object object at ...> != <object object at ...>'
)
def test_different_types(self):
self.check_raises('x', 1, "'x' != 1")
def test_number_same(self):
compare(1, 1)
def test_number_different(self):
self.check_raises(1, 2, '1 != 2')
def test_decimal_different(self):
self.check_raises(Decimal(1), Decimal(2),
"Decimal('1') != Decimal('2')")
def test_different_with_labels(self):
self.check_raises(1, 2, '1 (expected) != 2 (actual)',
x_label='expected', y_label='actual')
def test_string_same(self):
compare('x', 'x')
def test_unicode_string_different(self):
self.check_raises('a', b'b', "'a' != b'b'")
def test_bytes_different(self):
expected = (
"\n"
"b'12345678901'\n"
'!=\n'
"b'12345678902'"
)
self.check_raises(
b'12345678901',
b'12345678902',
expected
)
def test_bytes_same_strict(self):
compare(actual=b'', expected=b'', strict=True)
def test_moar_bytes_different(self):
self.check_raises(
actual=b'{"byte_pound":"b\'\\\\xa3\'"}',
expected=b'{"byte_pound":"b\\\'\\xa3\'"}',
message = (
"\n"
"b'{\"byte_pound\":\"b\\\\\\'\\\\xa3\\\'\"}' (expected)\n"
'!=\n'
"b'{\"byte_pound\":\"b\\\'\\\\\\\\xa3\\\'\"}' (actual)"
)
)
def test_string_diff_short(self):
self.check_raises(
'\n'+('x'*9), '\n'+('y'*9),
"'\\nxxxxxxxxx' != '\\nyyyyyyyyy'"
)
def test_string_diff_long(self):
self.check_raises(
'x'*11, 'y'*11,
"\n'xxxxxxxxxxx'\n!=\n'yyyyyyyyyyy'"
)
def test_string_diff_long_newlines(self):
self.check_raises(
'x'*5+'\n'+'y'*5, 'x'*5+'\n'+'z'*5,
"\n--- first\n+++ second\n@@ -1,2 +1,2 @@\n xxxxx\n-yyyyy\n+zzzzz"
)
def test_string_diff_short_labels(self):
self.check_raises(
'\n'+('x'*9), '\n'+('y'*9),
"'\\nxxxxxxxxx' (expected) != '\\nyyyyyyyyy' (actual)",
x_label='expected',
y_label='actual'
)
def test_string_diff_long_labels(self):
self.check_raises(
'x'*11, 'y'*11,
"\n'xxxxxxxxxxx' (expected)\n!=\n'yyyyyyyyyyy' (actual)",
x_label='expected',
y_label='actual'
)
def test_string_diff_long_newlines_labels(self):
self.check_raises(
'x'*5+'\n'+'y'*5, 'x'*5+'\n'+'z'*5,
"\n--- expected\n+++ actual\n"
"@@ -1,2 +1,2 @@\n xxxxx\n-yyyyy\n+zzzzz",
x_label='expected',
y_label='actual'
)
def test_exception_same_object(self):
e = ValueError('some message')
compare(e, e)
def test_exception_same_c_wrapper(self):
e1 = ValueError('some message')
e2 = ValueError('some message')
compare(C(e1), e2)
def test_exception_different_object(self):
e1 = ValueError('some message')
e2 = ValueError('some message')
compare(e1, e2)
def test_exception_different_object_c_wrapper(self):
e1 = ValueError('some message')
e2 = ValueError('some message')
compare(C(e1), e2)
def test_exception_diff(self):
e1 = ValueError('some message')
e2 = ValueError('some other message')
if PY_37_PLUS:
self.check_raises(
e1, e2,
"ValueError('some message') != ValueError('some other message')"
)
else:
self.check_raises(
e1, e2,
"ValueError('some message',) != ValueError('some other message',)"
)
def test_exception_diff_c_wrapper(self):
e1 = ValueError('some message')
e2 = ValueError('some other message')
self.check_raises(
C(e1), e2,
("\n"
"<C:builtins.ValueError(failed)>\n"
"attributes differ:\n"
"'args': ('some message',) (Comparison) "
"!= ('some other message',) (actual)\n"
"</C:builtins.ValueError>"
" != ValueError('some other message'{message})"
).format(message='' if PY_37_PLUS else ','))
def test_sequence_long(self):
self.check_raises(
['quite a long string 1', 'quite a long string 2',
'quite a long string 3', 'quite a long string 4',
'quite a long string 5', 'quite a long string 6',
'quite a long string 7', 'quite a long string 8'],
['quite a long string 1', 'quite a long string 2',
'quite a long string 3', 'quite a long string 4',
'quite a long string 9', 'quite a long string 10',
'quite a long string 11', 'quite a long string 12'],
"sequence not as expected:\n\n"
"same:\n"
"['quite a long string 1',\n"
" 'quite a long string 2',\n"
" 'quite a long string 3',\n"
" 'quite a long string 4']\n\n"
"first:\n"
"['quite a long string 5',\n"
" 'quite a long string 6',\n"
" 'quite a long string 7',\n"
" 'quite a long string 8']\n\n"
"second:\n"
"['quite a long string 9',\n"
" 'quite a long string 10',\n"
" 'quite a long string 11',\n"
" 'quite a long string 12']\n"
"\n"
"While comparing [4]: \n"
"'quite a long string 5'\n"
"!=\n"
"'quite a long string 9'"
)
def test_sequence_different_labels_supplied(self):
self.check_raises(
[1, 2, 3], [1, 2, 4],
"sequence not as expected:\n\n"
"same:\n"
"[1, 2]\n\n"
"expected:\n"
"[3]\n\n"
"actual:\n"
"[4]",
x_label='expected',
y_label='actual',
)
def test_list_same(self):
compare([1, 2, 3], [1, 2, 3])
def test_list_different(self):
self.check_raises(
[1, 2, 3], [1, 2, 4],
"sequence not as expected:\n\n"
"same:\n"
"[1, 2]\n\n"
"first:\n"
"[3]\n\n"
"second:\n"
"[4]"
)
def test_list_different_float(self):
self.check_raises(
[1, 2, 3.0], [1, 2, 4.0],
"sequence not as expected:\n\n"
"same:\n"
"[1, 2]\n\n"
"first:\n"
"[3.0]\n\n"
"second:\n"
"[4.0]"
)
def test_list_different_decimal(self):
self.check_raises(
[1, 2, Decimal(3)], [1, 2, Decimal(4)],
"sequence not as expected:\n\n"
"same:\n"
"[1, 2]\n\n"
"first:\n"
"[Decimal('3')]\n\n"
"second:\n"
"[Decimal('4')]"
)
def test_list_totally_different(self):
self.check_raises(
[1], [2],
"sequence not as expected:\n\n"
"same:\n"
"[]\n\n"
"first:\n"
"[1]\n\n"
"second:\n"
"[2]"
)
def test_list_first_shorter(self):
self.check_raises(
[1, 2], [1, 2, 3],
"sequence not as expected:\n\n"
"same:\n[1, 2]\n\n"
"first:\n[]\n\n"
"second:\n[3]"
)
def test_list_second_shorter(self):
self.check_raises(
[1, 2, 3], [1, 2],
"sequence not as expected:\n\n"
"same:\n[1, 2]\n\n"
"first:\n[3]\n\n"
"second:\n[]"
)
def test_dict_same(self):
compare(dict(x=1), dict(x=1))
def test_dict_first_missing_keys(self):
self.check_raises(
dict(), dict(z=3),
"dict not as expected:\n"
"\n"
"in second but not first:\n"
"'z': 3"
)
def test_dict_second_missing_keys(self):
self.check_raises(
dict(z=3), dict(),
"dict not as expected:\n"
"\n"
"in first but not second:\n"
"'z': 3"
)
def test_dict_values_different(self):
self.check_raises(
dict(x=1), dict(x=2),
"dict not as expected:\n"
"\n"
"values differ:\n"
"'x': 1 != 2"
)
def test_dict_identical_non_matching_ints(self):
self.check_raises(
dict(x=1, y=1), dict(x=2, y=2),
"dict not as expected:\n"
"\n"
"values differ:\n"
"'x': 1 != 2\n"
"'y': 1 != 2"
)
def test_dict_identical_non_matching_floats(self):
self.check_raises(
dict(x=1.0, y=1.0), dict(x=2.0, y=2.0),
"dict not as expected:\n"
"\n"
"values differ:\n"
"'x': 1.0 != 2.0\n"
"'y': 1.0 != 2.0"
)
def test_dict_identical_none_matching_nones_and_ones(self):
self.check_raises(
{
'foo': None,
'baz': None,
},
{
'foo': 1,
'baz': 1,
},
"dict not as expected:\n"
"\n"
'values differ:\n'
"'baz': None != 1\n"
"'foo': None != 1\n"
'\n'
"While comparing ['baz']: None != 1\n"
"\n"
"While comparing ['foo']: None != 1"
)
def test_dict_labels_specified(self):
self.check_raises(
dict(x=1, y=2), dict(x=2, z=3),
"dict not as expected:\n"
"\n"
"in expected but not actual:\n"
"'y': 2\n"
"\n"
"in actual but not expected:\n"
"'z': 3\n"
"\n"
"values differ:\n"
"'x': 1 (expected) != 2 (actual)",
x_label='expected',
y_label='actual'
)
def test_dict_tuple_keys_same_value(self):
compare({(1, 2): None}, {(1, 2): None})
def test_dict_tuple_keys_different_value(self):
self.check_raises(
{(1, 2): 3},
{(1, 2): 42},
"dict not as expected:\n"
"\n"
"values differ:\n"
"(1, 2): 3 != 42"
)
def test_dict_full_diff(self):
self.check_raises(
dict(x=1, y=2, a=4), dict(x=1, z=3, a=5),
"dict not as expected:\n"
"\n"
'same:\n'
"['x']\n"
"\n"
"in first but not second:\n"
"'y': 2\n"
'\n'
"in second but not first:\n"
"'z': 3\n"
'\n'
"values differ:\n"
"'a': 4 != 5"
)
def test_dict_consistent_ordering(self):
self.check_raises(
dict(xa=1, xb=2, ya=1, yb=2, aa=3, ab=4),
dict(xa=1, xb=2, za=3, zb=4, aa=5, ab=5),
"dict not as expected:\n"
"\n"
'same:\n'
"['xa', 'xb']\n"
"\n"
"in first but not second:\n"
"'ya': 1\n"
"'yb': 2\n"
'\n'
"in second but not first:\n"
"'za': 3\n"
"'zb': 4\n"
'\n'
"values differ:\n"
"'aa': 3 != 5\n"
"'ab': 4 != 5"
)
def test_dict_consistent_ordering_types_same(self):
self.check_raises(
{None: 1, 6: 2, 1: 3},
{None: 1, 6: 2, 1: 4},
"dict not as expected:\n"
"\n"+
'same:\n'+
"[6, None]\n"+
"\n"
"values differ:\n"
"1: 3 != 4"
)
def test_dict_consistent_ordering_types_x_not_y(self):
self.check_raises(
{None: 1, 3: 2},
{},
"dict not as expected:\n"
"\n"
"in first but not second:\n"
"3: 2\n"
"None: 1"
)
def test_dict_consistent_ordering_types_y_not_x(self):
self.check_raises(
{},
{None: 1, 3: 2},
"dict not as expected:\n"
"\n"
"in second but not first:\n"
"3: 2\n"
"None: 1"
)
def test_dict_consistent_ordering_types_value(self):
self.check_raises(
{None: 1, 6: 2},
{None: 3, 6: 4},
"dict not as expected:\n"
"\n"
"values differ:\n"
"6: 2 != 4\n"
"None: 1 != 3"
)
def test_set_same(self):
compare(set([1]), set([1]))
def test_set_first_missing_keys(self):
self.check_raises(
set(), set([3]),
"set not as expected:\n"
"\n"
"in second but not first:\n"
"[3]\n"
'\n'
)
def test_set_second_missing_keys(self):
self.check_raises(
set([3]), set(),
"set not as expected:\n"
"\n"
"in first but not second:\n"
"[3]\n"
'\n'
)
def test_set_full_diff(self):
self.check_raises(
set([1, 2, 4]), set([1, 3, 5]),
"set not as expected:\n"
"\n"
"in first but not second:\n"
"[2, 4]\n"
'\n'
"in second but not first:\n"
"[3, 5]\n"
'\n'
)
def test_set_type_ordering(self):
self.check_raises(
{None, 1}, {'', 2},
"set not as expected:\n"
"\n"
"in first but not second:\n"
"[1, None]\n"
'\n'
"in second but not first:\n"
"['', 2]\n"
'\n'
)
def test_set_labels(self):
self.check_raises(
set([1, 2, 4]), set([1, 3, 5]),
"set not as expected:\n"
"\n"
"in expected but not actual:\n"
"[2, 4]\n"
'\n'
"in actual but not expected:\n"
"[3, 5]\n"
'\n',
x_label='expected',
y_label='actual',
)
def test_tuple_same(self):
compare((1, 2, 3), (1, 2, 3))
def test_tuple_different(self):
self.check_raises(
(1, 2, 3), (1, 2, 4),
"sequence not as expected:\n\n"
"same:\n(1, 2)\n\n"
"first:\n(3,)\n\n"
"second:\n(4,)"
)
def test_tuple_totally_different(self):
self.check_raises(
(1, ), (2, ),
"sequence not as expected:\n\n"
"same:\n()\n\n"
"first:\n(1,)\n\n"
"second:\n(2,)"
)
def test_tuple_first_shorter(self):
self.check_raises(
(1, 2), (1, 2, 3),
"sequence not as expected:\n\n"
"same:\n(1, 2)\n\n"
"first:\n()\n\n"
"second:\n(3,)"
)
def test_tuple_second_shorter(self):
self.check_raises(
(1, 2, 3), (1, 2),
"sequence not as expected:\n\n"
"same:\n(1, 2)\n\n"
"first:\n(3,)\n\n"
"second:\n()"
)
def test_generator_same(self):
compare(generator(1, 2, 3), generator(1, 2, 3))
def test_generator_different(self):
self.check_raises(
generator(1, 2, 3), generator(1, 2, 4),
"sequence not as expected:\n\n"
"same:\n(1, 2)\n\n"
"first:\n(3,)\n\n"
"second:\n(4,)"
)
def test_generator_totally_different(self):
self.check_raises(
generator(1, ), generator(2, ),
"sequence not as expected:\n\n"
"same:\n()\n\n"
"first:\n(1,)\n\n"
"second:\n(2,)"
)
def test_generator_first_shorter(self):
self.check_raises(
generator(1, 2), generator(1, 2, 3),
"sequence not as expected:\n\n"
"same:\n(1, 2)\n\n"
"first:\n()\n\n"
"second:\n(3,)"
)
def test_generator_second_shorted(self):
self.check_raises(
generator(1, 2, 3), generator(1, 2),
"sequence not as expected:\n\n"
"same:\n(1, 2)\n\n"
"first:\n(3,)\n\n"
"second:\n()"
)
def test_nested_generator_different(self):
self.check_raises(
generator(1, 2, generator(3), 4),
generator(1, 2, generator(3), 5),
"sequence not as expected:\n"
"\n"
"same:\n"
"(1, 2, <generator object generator at ...>)\n"
"\n"
"first:\n"
"(4,)\n"
"\n"
"second:\n"
"(5,)"
)
def test_nested_generator_tuple_left(self):
compare(
generator(1, 2, (3, ), 4),
generator(1, 2, generator(3), 4),
)
def test_nested_generator_tuple_right(self):
compare(
generator(1, 2, generator(3), 4),
generator(1, 2, (3, ), 4),
)
def test_sequence_and_generator(self):
compare((1, 2, 3), generator(1, 2, 3))
def test_sequence_and_generator_strict(self):
expected = compile(
"\(1, 2, 3\) \(<(class|type) 'tuple'>\) \(expected\) != "
"<generator object (generator )?at... "
"\(<(class|type) 'generator'>\) \(actual\)"
)
self.check_raises(
(1, 2, 3), generator(1, 2, 3),
regex=expected,
strict=True,
x_label='expected',
y_label='actual',
)
def test_generator_and_sequence(self):
compare(generator(1, 2, 3), (1, 2, 3))
def test_iterable_with_iterable_same(self):
compare(range(1, 4), range(1, 4))
def test_iterable_with_iterable_different(self):
self.check_raises(
range(1, 4), range(1, 3),
"sequence not as expected:\n"
"\n"
"same:\n"
"(1, 2)\n"
"\n"
"first:\n"
"(3,)\n"
"\n"
"second:\n"
"()"
)
def test_iterable_and_generator(self):
compare(range(1, 4), generator(1, 2, 3))
def test_iterable_and_generator_strict(self):
expected = compile(
"range\(1, 4\) \(<(class|type) 'x?range'>\) != "
"<generator object (generator )?at... "
"\(<(class|type) 'generator'>\)"
)
self.check_raises(
range(1, 4), generator(1, 2, 3),
regex=expected,
strict=True,
)
def test_generator_and_iterable(self):
compare(generator(1, 2, 3), range(1, 4))
def test_tuple_and_list(self):
compare((1, 2, 3), [1, 2, 3])
def test_tuple_and_list_strict(self):
expected = ("(1, 2, 3) (<class 'tuple'>) != "
"[1, 2, 3] (<class 'list'>)")
self.check_raises(
(1, 2, 3), [1, 2, 3],
expected,
strict=True
)
def test_float_subclass_strict(self):
class TestFloat(float):
pass
compare(TestFloat(0.75), TestFloat(0.75), strict=True)
def test_old_style_classes_same(self):
class X:
pass
compare(X, X)
def test_default_style_classes_different(self):
expected = (
"<class 'testfixtures.tests.test_compare.TestCompare."
"test_default_style_classes_different.<locals>.X'>"
" != "
"<class 'testfixtures.tests.test_compare.TestCompare."
"test_default_style_classes_different.<locals>.Y'>"
)
class X:
pass
class Y:
pass
self.check_raises(X, Y, expected)
def test_new_style_classes_same(self):
class X(object):
pass
compare(X, X)
def test_new_style_classes_different(self):
expected = (
"<class 'testfixtures.tests.test_compare.TestCompare."
"test_new_style_classes_different.<locals>.X'>"
" != "
"<class 'testfixtures.tests.test_compare.TestCompare."
"test_new_style_classes_different.<locals>.Y'>"
)
class X(object):
pass
class Y(object):
pass
self.check_raises(X, Y, expected)
def test_show_whitespace(self):
# does nothing! ;-)
self.check_raises(
' x \n\r', ' x \n \t',
"' x \\n\\r' != ' x \\n \\t'",
show_whitespace=True
)
def test_show_whitespace_long(self):
self.check_raises(
"\t \n '", '\r \n ',
'\n--- first\n'
'+++ second\n'
'@@ -1,2 +1,2 @@\n'
'-\'\\t \\n\'\n'
'-" \'"\n'
'+\'\\r \\n\'\n'
'+\' \'',
show_whitespace=True
)
def test_show_whitespace_equal(self):
compare('x', 'x', show_whitespace=True)
def test_show_whitespace_not_used_because_of_other_difference(self):
self.check_raises(
(1, 'a'),
(2, 'b'),
"sequence not as expected:\n"
"\n"
"same:\n"
"()\n"
"\n"
"first:\n"
"(1, 'a')\n"
"\n"
"second:\n"
"(2, 'b')",
show_whitespace=False
)
def test_include_trailing_whitespace(self):
self.check_raises(
' x \n', ' x \n',
"' x \\n' != ' x \\n'"
)
def test_ignore_trailing_whitespace(self):
compare(' x \t\n', ' x\t \n', trailing_whitespace=False)
def test_ignore_trailing_whitespace_non_string(self):
self.check_raises(
1, '',
"1 != ''",
trailing_whitespace=False
)
def test_ignore_trailing_whitespace_but_respect_leading_whitespace(self):
# NB: careful: this strips off the last newline too
# DON'T use if you care about that!
self.check_raises(
'a\n b\n c\n',
'a\nb\nc\n',
"'a\\n b\\n c' != 'a\\nb\\nc'",
trailing_whitespace=False
)
def test_include_blank_lines(self):
self.check_raises(
'\n \n', '\n ',
"'\\n \\n' != '\\n '"
)
def test_ignore_blank_lines(self):
compare("""
a
\t
b
""",
' a\nb', blanklines=False)
def test_ignore_blank_lines_non_string(self):
self.check_raises(
1, '',
"1 != ''",
blanklines=False
)
def test_supply_comparer(self):
def compare_dict(x, y, context):
self.assertEqual(x, {1: 1})
self.assertEqual(y, {2: 2})
self.assertEqual(context.get_option('foo'), 'bar')
return 'not equal'
with ShouldAssert('not equal'):
compare({1: 1}, {2: 2},
foo='bar',
comparers={dict: compare_dict})
def test_register_more_specific(self):
class_ = namedtuple('Test', 'x')
with ShouldAssert('compare class_'):
compare(class_(1), class_(2),
comparers={
tuple: Mock(return_value='compare tuple'),
class_: Mock(return_value='compare class_')
})
def test_extra_comparers_leave_existing(self):
class MyObject(object):
def __init__(self, name):
self.name = name
def __repr__(self):
return 'MyObject instance'
def compare_my_object(x, y, context):
return '%s != %s' % (x.name, y.name)
with Replacer() as r:
r.replace('testfixtures.comparison._registry', {
list: compare_sequence,
})
self.check_raises(
[1, MyObject('foo')], [1, MyObject('bar')],
"sequence not as expected:\n"
"\n"
"same:\n"
"[1]\n"
"\n"
"first:\n"
"[MyObject instance]\n"
"\n"
"second:\n"
"[MyObject instance]\n"
"\n"
"While comparing [1]: foo != bar",
comparers={MyObject: compare_my_object}
)
def test_list_subclass(self):
class MyList(list): pass
a_list = MyList([1])
b_list = MyList([2])
self.check_raises(
a_list, b_list,
"sequence not as expected:\n\n"
"same:\n[]\n\n"
"first:\n[1]\n\n"
"second:\n[2]"
)
def test_same_object_strict_okay(self):
m = object()
compare(m, m, strict=True)
def test_strict_comparer_supplied(self):
compare_obj = Mock()
compare_obj.return_value = 'not equal'
self.check_raises(
object(), object(),
"not equal",
strict=True,
comparers={object: compare_obj},
)
def test_strict_default_comparer(self):
class MyList(list):
pass
# default comparer used!
self.check_raises(
MyList((1, 2, 3)), MyList((1, 2, 4)),
"sequence not as expected:\n"
"\n"
"same:\n"
"[1, 2]\n"
"\n"
"first:\n"
"[3]\n"
"\n"
"second:\n"
"[4]",
strict=True,
)
def test_list_subclass_strict(self):
m = Mock()
m.aCall()
self.check_raises(
[call.aCall()], m.method_calls,
("[call.aCall()] (<class 'list'>) != [call.aCall()] "
f"({call_list_repr})"),
strict=True,
)
def test_list_subclass_long_strict(self):
m = Mock()
m.call('X'*20)
self.check_raises(
[call.call('Y'*20)], m.method_calls,
("[call.call('YYYYYYYYYYYYYYYYYY... "
"(<class 'list'>) != "
"[call.call('XXXXXXXXXXXXXXXXXX... "
f"({call_list_repr})"),
strict=True,
)
def test_prefix(self):
self.check_raises(1, 2, 'wrong number of orders: 1 != 2',
prefix='wrong number of orders')
def test_prefix_multiline(self):
self.check_raises(
'x'*5+'\n'+'y'*5, 'x'*5+'\n'+'z'*5,
"file content: \n--- first\n+++ second\n"
"@@ -1,2 +1,2 @@\n xxxxx\n-yyyyy\n+zzzzz",
prefix='file content'
)
def test_prefix_callable(self):
with ShouldAssert('foo: 1 != 2'):
compare(1, 2, prefix=lambda: 'foo')
def test_prefix_stringable(self):
with ShouldAssert('foo: 1 != 2'):
compare(1, 2, prefix=Lazy('foo'))
def test_prefix_lazy(self):
compare(2, 2, prefix=Mock(side_effect=Exception('boom!')))
def test_suffix(self):
self.check_raises(
1, 2,
'1 != 2\n'
'additional context',
suffix='additional context',
)
def test_suffix_callable(self):
with ShouldAssert('1 != 2\n3'):
compare(1, 2, suffix=lambda: 3)
def test_suffix_stringable(self):
with ShouldAssert('1 != 2\nfoo'):
compare(1, 2, suffix=Lazy('foo'))
def test_suffix_lazy(self):
compare(2, 2, suffix=Mock(side_effect=Exception('boom!')))
def test_labels_multiline(self):
self.check_raises(
'x'*5+'\n'+'y'*5, 'x'*5+'\n'+'z'*5,
"\n--- expected\n+++ actual\n"
"@@ -1,2 +1,2 @@\n xxxxx\n-yyyyy\n+zzzzz",
x_label='expected',
y_label='actual'
)
def test_generator_with_non_generator(self):
self.check_raises(
generator(1, 2, 3), None,
'<generator object generator at ...> != None',
)
def test_generator_with_buggy_generator(self):
def bad_gen():
yield 1
# raising a TypeError here is important :-/
raise TypeError('foo')
with ShouldRaise(TypeError('foo')):
compare(generator(1, 2, 3), bad_gen())
def test_nested_dict_tuple_values_different(self):
self.check_raises(
dict(x=(1, 2, 3)), dict(x=(1, 2, 4)),
"dict not as expected:\n"
"\n"
"values differ:\n"
"'x': (1, 2, 3) != (1, 2, 4)\n"
'\n'
"While comparing ['x']: sequence not as expected:\n"
"\n"
"same:\n"
"(1, 2)\n"
"\n"
"first:\n"
"(3,)\n"
"\n"
"second:\n"
"(4,)"
)
def test_nested_dict_different(self):
self.check_raises(
dict(x=dict(y=1)), dict(x=dict(y=2)),
"dict not as expected:\n"
"\n"
"values differ:\n"
"'x': {'y': 1} != {'y': 2}\n"
'\n'
"While comparing ['x']: dict not as expected:\n"
"\n"
"values differ:\n"
"'y': 1 != 2"
)
def test_nested_dict_empty_but_same(self):
compare(dict(x=dict()), dict(x=dict()), ignore_eq=True)
def test_nested_dict_empty_with_keys(self):
compare(dict(x=dict(x=1)), dict(x=dict(x=1)), ignore_eq=True)
def test_tuple_list_different(self):
self.check_raises(
(1, [2, 3, 5]), (1, [2, 4, 5]),
"sequence not as expected:\n"
"\n"
"same:\n"
"(1,)\n"
"\n"
"first:\n"
"([2, 3, 5],)\n"
"\n"
"second:\n"
"([2, 4, 5],)\n"
"\n"
"While comparing [1]: sequence not as expected:\n"
"\n"
"same:\n"
"[2]\n"
"\n"
"first:\n"
"[3, 5]\n"
"\n"
"second:\n"
"[4, 5]"
)
def test_tuple_long_strings_different(self):
self.check_raises(
(1, 2, "foo\nbar\nbaz\n", 4),
(1, 2, "foo\nbob\nbaz\n", 4),
"sequence not as expected:\n"
"\n"
"same:\n"
"(1, 2)\n"
"\n"
"first:\n"
"('foo\\nbar\\nbaz\\n', 4)\n"
"\n"
"second:\n"
"('foo\\nbob\\nbaz\\n', 4)\n"
"\n"
"While comparing [2]: \n"
"--- first\n"
"+++ second\n"
"@@ -1,4 +1,4 @@\n"
# check that show_whitespace bubbles down
" 'foo\\n'\n"
"-'bar\\n'\n"
"+'bob\\n'\n"
" 'baz\\n'\n"
" ''",
show_whitespace=True
)
def test_dict_multiple_differences(self):
self.check_raises(
dict(x=(1, 2, 3), y=(4, 5, 6, )),
dict(x=(1, 2, 4), y=(4, 5, 7, )),
"dict not as expected:\n"
"\n"
"values differ:\n"
"'x': (1, 2, 3) != (1, 2, 4)\n"
"'y': (4, 5, 6) != (4, 5, 7)\n"
"\n"
"While comparing ['x']: sequence not as expected:\n"
"\n"
"same:\n"
"(1, 2)\n"
"\n"
"first:\n"
"(3,)\n"
"\n"
"second:\n"
"(4,)\n"
"\n"
"While comparing ['y']: sequence not as expected:\n"
"\n"
"same:\n"
"(4, 5)\n"
"\n"
"first:\n"
"(6,)\n"
"\n"
"second:\n"
"(7,)"
)
def test_deep_breadcrumbs(self):
obj1 = singleton('obj1')
obj2 = singleton('obj2')
gen1 = generator(obj1, obj2)
gen2 = generator(obj1, )
# dict -> list -> tuple -> generator
self.check_raises(
dict(x=[1, ('a', 'b', gen1), 3], y=[3, 4]),
dict(x=[1, ('a', 'b', gen2), 3], y=[3, 4]), (
"dict not as expected:\n"
"\n"
"same:\n"
"['y']\n"
"\n"
"values differ:\n"
"'x': [1, ('a', 'b', {gen1}), 3] != [1, ('a', 'b', {gen2}), 3]"
"\n\n"
"While comparing ['x']: sequence not as expected:\n"
"\n"
"same:\n"
"[1]\n"
"\n"
"first:\n"
"[('a', 'b', {gen1}), 3]\n"
"\n"
"second:\n"
"[('a', 'b', {gen2}), 3]\n"
"\n"
"While comparing ['x'][1]: sequence not as expected:\n"
"\n"
"same:\n"
"('a', 'b')\n"
"\n"
"first:\n"
"({gen1},)\n"
"\n"
"second:\n"
"({gen2},)\n"
"\n"
"While comparing ['x'][1][2]: sequence not as expected:\n"
"\n"
"same:\n"
"(<obj1>,)\n"
"\n"
"first:\n"
"(<obj2>,)\n"
"\n"
"second:\n"
"()"
).format(gen1=hexsub(repr(gen1)),
gen2=hexsub(repr(gen2)))
)
def test_nested_labels(self):
obj1 = singleton('obj1')
obj2 = singleton('obj2')
gen1 = generator(obj1, obj2)
gen2 = generator(obj1, )
# dict -> list -> tuple -> generator
self.check_raises(
dict(x=[1, ('a', 'b', gen1), 3], y=[3, 4]),
dict(x=[1, ('a', 'b', gen2), 3], y=[3, 4]), (
"dict not as expected:\n"
"\n"
"same:\n"
"['y']\n"
"\n"
"values differ:\n"
"'x': [1, ('a', 'b', {gen1}), 3] (expected) != "
"[1, ('a', 'b', {gen2}), 3] (actual)\n"
"\n"
"While comparing ['x']: sequence not as expected:\n"
"\n"
"same:\n"
"[1]\n"
"\n"
"expected:\n"
"[('a', 'b', {gen1}), 3]\n"
"\n"
"actual:\n"
"[('a', 'b', {gen2}), 3]\n"
"\n"
"While comparing ['x'][1]: sequence not as expected:\n"
"\n"
"same:\n"
"('a', 'b')\n"
"\n"
"expected:\n"
"({gen1},)\n"
"\n"
"actual:\n"
"({gen2},)\n"
"\n"
"While comparing ['x'][1][2]: sequence not as expected:\n"
"\n"
"same:\n"
"(<obj1>,)\n"
"\n"
"expected:\n"
"(<obj2>,)\n"
"\n"
"actual:\n"
"()"
).format(gen1=hexsub(repr(gen1)),
gen2=hexsub(repr(gen2))),
x_label='expected',
y_label='actual',
)
def test_nested_strict_only_type_difference(self):
MyTuple = namedtuple('MyTuple', 'x y z')
type_repr = repr(MyTuple)
tuple_repr = repr(tuple)
self.check_raises(
[MyTuple(1, 2, 3)],
[(1, 2, 3)],
("sequence not as expected:\n"
"\n"
"same:\n"
"[]\n"
"\n"
"first:\n"
"[MyTuple(x=1, y=2, z=3)]\n"
"\n"
"second:\n"
"[(1, 2, 3)]\n"
"\n"
"While comparing [0]: MyTuple(x=1, y=2, z=3) "
"(%s) "
"!= (1, 2, 3) "
"(%s)") % (type_repr, tuple_repr),
strict=True
)
def test_strict_nested_different(self):
expected = "[1, 2] (<class 'list'>) != (1, 3) (<class 'tuple'>)"
self.check_raises(
(1, 2, [1, 2]), (1, 2, (1, 3)),
"sequence not as expected:\n"
"\n"
"same:\n"
"(1, 2)\n"
"\n"
"first:\n"
"([1, 2],)\n"
"\n"
"second:\n"
"((1, 3),)"
"\n\n"
"While comparing [2]: " + expected,
strict=True,
)
def test_namedtuple_equal(self):
class_ = namedtuple('Foo', 'x')
compare(class_(1), class_(1))
def test_namedtuple_same_type(self):
class_ = namedtuple('Foo', 'x y')
self.check_raises(
class_(1, 2), class_(1, 3),
"Foo not as expected:\n\n"
"same:\n"
"['x']\n\n"
"values differ:\n"
"'y': 2 != 3"
)
def test_namedtuple_different_type(self):
class_a = namedtuple('Foo', 'x y')
class_b = namedtuple('Bar', 'x y z')
self.check_raises(
class_a(1, 2), class_b(1, 2, 3),
"Foo(x=1, y=2) (<class 'testfixtures.tests.test_compare.Foo'>) != "
"Bar(x=1, y=2, z=3) "
"(<class 'testfixtures.tests.test_compare.Bar'>)"
)
def test_dict_with_list(self):
self.check_raises(
{1: 'one', 2: 'two'}, [1, 2],
"{1: 'one', 2: 'two'} != [1, 2]"
)
def test_explicit_expected(self):
self.check_raises('x', expected='y',
message="'y' (expected) != 'x' (actual)")
def test_explicit_actual(self):
self.check_raises('x', actual='y',
message="'x' (expected) != 'y' (actual)")
def test_explicit_both(self):
self.check_raises(expected='x', actual='y',
message="'x' (expected) != 'y' (actual)")
def test_implicit_and_labels(self):
self.check_raises('x', 'y',
x_label='x_label', y_label='y_label',
message="'x' (x_label) != 'y' (y_label)")
def test_explicit_and_labels(self):
self.check_raises(x='x', y='y',
x_label='x_label', y_label='y_label',
message="'x' (x_label) != 'y' (y_label)")
def test_invalid_two_args_expected(self):
with ShouldRaise(TypeError(
"Exactly two objects needed, you supplied: ['z', 'x', 'y']"
)):
compare('x', 'y', expected='z')
def test_invalid_two_args_actual(self):
with ShouldRaise(TypeError(
"Exactly two objects needed, you supplied: ['x', 'y', 'z']"
)):
compare('x', 'y', actual='z')
def test_invalid_zero_args(self):
with ShouldRaise(TypeError(
'Exactly two objects needed, you supplied:'
)):
compare()
def test_invalid_one_args(self):
with ShouldRaise(TypeError(
"Exactly two objects needed, you supplied: ['x']"
)):
compare('x')
def test_invalid_three_args(self):
with ShouldRaise(TypeError(
"Exactly two objects needed, you supplied: ['x', 'y', 'z']"
)):
compare('x', 'y', 'z')
def test_invalid_because_of_typo(self):
with ShouldRaise(TypeError(
"Exactly two objects needed, you supplied: ['x'] {'expceted': 'z'}"
)):
compare('x', expceted='z')
def test_dont_raise(self):
self.assertEqual(compare('x', 'y', raises=False), "'x' != 'y'")
class OrmObj(object):
def __init__(self, a):
self.a = a
def __eq__(self, other):
return True
def __repr__(self):
return 'OrmObj: '+str(self.a)
def test_django_orm_is_horrible(self):
self.assertTrue(self.OrmObj(1) == self.OrmObj(2))
def query_set():
yield self.OrmObj(1)
yield self.OrmObj(2)
self.check_raises(
message=(
"sequence not as expected:\n"
"\n"
"same:\n"
"(OrmObj: 1,)\n"
"\n"
"expected:\n"
"(OrmObj: 3,)\n"
"\n"
"actual:\n"
"(OrmObj: 2,)\n"
'\n'
'While comparing [1]: OrmObj not as expected:\n'
'\n'
'attributes differ:\n'
"'a': 3 (expected) != 2 (actual)"
),
expected=[self.OrmObj(1), self.OrmObj(3)],
actual=query_set(),
ignore_eq=True
)
def test_django_orm_is_horrible_part_2(self):
t_compare = partial(compare, ignore_eq=True)
t_compare(self.OrmObj(1), self.OrmObj(1))
t_compare(self.OrmObj('some longish string'),
self.OrmObj('some longish string'))
t_compare(self.OrmObj(date(2016, 1, 1)),
self.OrmObj(date(2016, 1, 1)))
def test_django_orm_is_horrible_part_3(self):
compare(
expected=self.OrmObj(1),
actual=self.OrmObj(1),
ignore_eq=True
)
def test_django_orm_is_horrible_part_4(self):
self.check_raises(
message='[1] (expected) != 2 (actual)',
expected=[1],
actual=2,
ignore_eq=True
)
def test_nested_django_orm_in_object(self):
class MyObject(object):
def __init__(self, orm):
self.orm = orm
self.check_raises(
message="MyObject not as expected:\n"
"\n"
"attributes differ:\n"
"'orm': OrmObj: 1 (expected) != OrmObj: 2 (actual)\n"
"\n"
"While comparing .orm: OrmObj not as expected:\n"
"\n"
"attributes differ:\n"
"'a': 1 (expected) != 2 (actual)",
expected=MyObject(self.OrmObj(1)),
actual=MyObject(self.OrmObj(2)),
ignore_eq=True)
def test_mock_call_same(self):
m = Mock()
m.foo(1, 2, x=3)
compare(m.mock_calls, m.mock_calls)
def test_mock_call_same_strict(self):
m = Mock()
m.foo(1, 2, x=3)
compare(m.mock_calls, m.mock_calls, strict=True)
def test_mock_call_equal(self):
m1 = Mock()
m1.foo(1, 2, x=3)
m2 = Mock()
m2.foo(1, 2, x=3)
compare(m1.mock_calls, m2.mock_calls)
def test_mock_call_equal_strict(self):
m1 = Mock()
m1.foo(1, 2, x=3)
m2 = Mock()
m2.foo(1, 2, x=3)
compare(m1.mock_calls, m2.mock_calls, strict=True)
def test_calls_different(self):
m1 = Mock()
m2 = Mock()
m1.foo(1, 2, x=3, y=4)
m2.bar(1, 3, x=7, y=4)
self.check_raises(
m1.mock_calls,
m2.mock_calls,
"sequence not as expected:\n"
"\n"
"same:\n"
"[]\n"
"\n"
"first:\n"
"[call.foo(1, 2, x=3, y=4)]\n"
"\n"
"second:\n"
"[call.bar(1, 3, x=7, y=4)]"
"\n\n"
'While comparing [0]: \n'
"'call.foo(1, 2, x=3, y=4)'\n"
'!=\n'
"'call.bar(1, 3, x=7, y=4)'"
)
def test_call_args_different(self):
m = Mock()
m.foo(1)
self.check_raises(
m.foo.call_args,
call(2),
"'call(1)' != 'call(2)'"
)
def test_calls_args_different_but_same_repr(self):
class Annoying(object):
def __init__(self, x):
self.x = x
def __repr__(self):
return '<annoying>'
m1 = Mock()
m2 = Mock()
m1.foo(Annoying(1))
m2.foo(Annoying(3))
self.check_raises(
m1.mock_calls,
m2.mock_calls,
'sequence not as expected:\n'
'\n'
'same:\n'
'[]\n'
'\n'
'first:\n'
'[call.foo(<annoying>)]\n'
'\n'
'second:\n'
'[call.foo(<annoying>)]\n'
'\n'
'While comparing [0]: mock.call not as expected:\n'
'\n'
'While comparing [0] args: sequence not as expected:\n'
'\n'
'same:\n'
'()\n'
'\n'
'first:\n'
'(<annoying>,)\n'
'\n'
'second:\n'
'(<annoying>,)\n'
'\n'
'While comparing [0] args[0]: Annoying not as expected:\n'
'\n'
'attributes differ:\n'
"'x': 1 != 3"
)
def test_calls_nested_equal_sub_attributes(self):
class Annoying(object):
def __init__(self, x):
self.x = x
def __repr__(self):
return '<annoying>'
m1 = Mock()
m2 = Mock()
m1.foo(x=[Annoying(1)])
m2.foo(x=[Annoying(1)])
compare(m1.mock_calls, m2.mock_calls)
def test_compare_arbitrary_nested_diff(self):
class OurClass:
def __init__(self, *args):
self.args = args
def __repr__(self):
return '<OurClass obj>'
self.check_raises(
OurClass(OurClass(1)),
OurClass(OurClass(2)),
"OurClass not as expected:\n"
"\n"
'attributes differ:\n'
"'args': (<OurClass obj>,) != (<OurClass obj>,)\n"
'\n'
'While comparing .args: sequence not as expected:\n'
'\n'
'same:\n'
'()\n'
'\n'
'first:\n'
'(<OurClass obj>,)\n'
'\n'
'second:\n'
'(<OurClass obj>,)\n'
'\n'
'While comparing .args[0]: OurClass not as expected:\n'
'\n'
'attributes differ:\n'
"'args': (1,) != (2,)\n"
'\n'
'While comparing .args[0].args: sequence not as expected:\n'
'\n'
'same:\n'
'()\n'
'\n'
'first:\n'
'(1,)\n'
'\n'
'second:\n'
'(2,)'
)
def test_compare_slotted_same(self):
compare(Slotted(1, 2), Slotted(1, 2))
def test_compare_slotted_diff(self):
self.check_raises(
Slotted(1, 2),
Slotted(1, 3),
"Slotted not as expected:\n"
"\n"
"attributes same:\n"
"['x']\n"
"\n"
'attributes differ:\n'
"'y': 2 != 3"
)
def test_empty_sets(self):
compare(set(), set())
def test_empty_sets_strict(self):
compare(set(), set(), strict=True)
def test_datetime_not_equal(self):
self.check_raises(
datetime(2001, 1, 1),
datetime(2001, 1, 2),
"datetime.datetime(2001, 1, 1, 0, 0) != "
"datetime.datetime(2001, 1, 2, 0, 0)"
)
def test_inherited_slots(self):
class Parent(object):
__slots__ = ('a',)
class Child(Parent):
__slots__ = ('b',)
def __init__(self, a, b):
self.a, self.b = a, b
self.check_raises(
Child(1, 'x'),
Child(2, 'x'),
'Child not as expected:\n'
'\n'
'attributes same:\n'
"['b']\n"
'\n'
'attributes differ:\n'
"'a': 1 != 2"
)
def test_empty_child_slots(self):
class Parent(object):
__slots__ = ('a',)
def __init__(self, a):
self.a = a
class Child(Parent):
__slots__ = ()
compare(Child(1), Child(1))
def test_slots_and_attrs(self):
class Parent(object):
__slots__ = ('a',)
class Child(Parent):
def __init__(self, a, b):
self.a = a
self.b = b
self.check_raises(Child(1, 2), Child(1, 3), message=(
'Child not as expected:\n'
'\n'
'attributes same:\n'
"['a']\n"
'\n'
'attributes differ:\n'
"'b': 2 != 3"
))
def test_partial_callable_different(self):
def foo(x): pass
def bar(y): pass
self.check_raises(
partial(foo),
partial(bar),
(
'partial not as expected:\n'
'\n'
'attributes same:\n'
"['args', 'keywords']\n"
'\n'
'attributes differ:\n'
"'func': {foo} != {bar}\n"
'\n'
'While comparing .func: {foo} != {bar}'
).format(foo=hexsub(repr(foo)), bar=hexsub(repr(bar))))
def test_partial_args_different(self):
def foo(x): pass
self.check_raises(
partial(foo, 1),
partial(foo, 2),
'partial not as expected:\n'
'\n'
'attributes same:\n'
"['func', 'keywords']\n"
'\n'
'attributes differ:\n'
"'args': (1,) != (2,)\n"
'\n'
'While comparing .args: sequence not as expected:\n'
'\n'
'same:\n'
'()\n'
'\n'
'first:\n'
'(1,)\n'
'\n'
'second:\n'
'(2,)'
)
def test_partial_kw_different(self):
def foo(x): pass
self.check_raises(
partial(foo, x=1, y=3),
partial(foo, x=2, z=4),
'partial not as expected:\n'
'\n'
'attributes same:\n'
"['args', 'func']\n"
'\n'
'attributes differ:\n'
"'keywords': {'x': 1, 'y': 3} != {'x': 2, 'z': 4}\n"
'\n'
'While comparing .keywords: dict not as expected:\n'
'\n'
'in first but not second:\n'
"'y': 3\n"
'\n'
'in second but not first:\n'
"'z': 4\n"
'\n'
'values differ:\n'
"'x': 1 != 2"
)
def test_partial_equal(self):
def foo(x): pass
compare(partial(foo, 1, x=2), partial(foo, 1, x=2))
def test_repr_and_attributes_equal(self):
class Wut(object):
def __repr__(self):
return 'Wut'
def __eq__(self, other):
return False
self.check_raises(
Wut(),
Wut(),
"Both x and y appear as 'Wut', but are not equal!"
)
self.check_raises(
expected=Wut(),
actual=Wut(),
message="Both expected and actual appear as 'Wut', but are not equal!"
)
def test_string_with_slotted(self):
class Slotted(object):
__slots__ = ['foo']
def __init__(self, foo):
self.foo = foo
def __repr__(self):
return repr(self.foo)
self.check_raises(
'foo',
Slotted('foo'),
"'foo' (%s) != 'foo' (%s)" % (repr(str), repr(Slotted))
)
def test_not_recursive(self):
self.check_raises(
{1: 'foo', 2: 'foo'},
{1: 'bar', 2: 'bar'},
"dict not as expected:\n"
"\n"
"values differ:\n"
"1: 'foo' != 'bar'\n"
"2: 'foo' != 'bar'\n"
"\n"
"While comparing [1]: 'foo' != 'bar'"
"\n\n"
"While comparing [2]: 'foo' != 'bar'"
)
def test_regex(self):
shared_prefix = "a" * 199
self.check_raises(
re.compile(shared_prefix + "x"),
re.compile(shared_prefix + "y"),
'Both x and y appear as "re.compile(\''+'a'*199+')", but are not equal!'
)
def test_self_referential_same(self):
expected = {1: 'foo'}
expected[2] = expected
actual = {1: 'foo'}
actual[2] = actual
compare(expected, actual)
def test_self_referential_different(self):
expected = {1: 'foo'}
expected[2] = expected
actual = {1: 'bar'}
actual[2] = actual
self.check_raises(
expected,
actual,
'dict not as expected:\n'
'\n'
'same:\n'
'[2]\n'
'\n'
'values differ:\n'
"1: 'foo' != 'bar'\n"
'\n'
"While comparing [1]: 'foo' != 'bar'"
)
def test_self_referential_different_but_shows_already_seen(self):
ouroboros1 = {}
ouroboros1['ouroboros'] = ouroboros1
ouroboros2 = {}
ouroboros2['ouroboros'] = ouroboros2
id2 = str(id(ouroboros2))
self.check_raises(
{1: ouroboros1, 2: 'foo'},
{1: ouroboros2, 2: ouroboros2},
'dict not as expected:\n'
'\n'
'same:\n'
'[1]\n'
'\n'
'values differ:\n'
"2: 'foo' != {'ouroboros': <Recursion on dict with id="+id2+">}\n"
'\n'
"While comparing [2]: 'foo' != "
"<AlreadySeen for {'ouroboros': {...}} at [1] with id "+id2+">"
)
def test_self_referential_object_tree(self):
class Node:
def __init__(self):
self.parent = None
self.children = []
def add(self, child: 'Node'):
self.children.append(child)
child.parent = self
def __repr__(self):
return f'<Node: {self.children}>'
expected = Node()
expected.add(Node())
expected.add(Node())
actual = Node()
actual.add(Node())
self.check_raises(
expected,
actual,
'Node not as expected:\n'
'\n'
'attributes same:\n'
"['parent']\n"
'\n'
'attributes differ:\n'
"'children': [<Node: []>, <Node: []>] != [<Node: []>]\n"
'\n'
'While comparing .children: sequence not as expected:\n'
'\n'
'same:\n'
'[<Node: []>]\n'
'\n'
'first:\n'
'[<Node: []>]\n'
'\n'
'second:\n'
'[]'
)
def test_repeated_object_on_the_left_side_ignore_eq(self):
item = [1, 2, 3]
compare(expected=[item, item], actual=[[1, 2, 3], [1, 2, 3]], ignore_eq=True)
def test_repeated_object_on_both_sides_ignore_eq(self):
item = [1, 2, 3]
compare(expected=[item, item], actual=[item, [1, 2, 3]], ignore_eq=True)
def test_repeated_object_on_both_sides_left_at_compare_strict_type_same(self):
item = [1, 2, 3]
compare(expected=[item, item], actual=[item, [1, 2, 3]], strict=True)
def test_repeated_object_on_both_sides_right_at_compare_strict_type_same(self):
item = [1, 2, 3]
compare(expected=[item, [1, 2, 3]], actual=[item, item], strict=True)
def test_repeated_object_on_both_sides_strict_type_different(self):
item = [1, 2, 3]
class MyList(list):
def __repr__(self):
return f'<{type(self).__name__}:{super().__repr__()}>'
type_repr = repr(MyList)
self.check_raises(
[item, item],
[item, MyList((1, 2, 3))],
strict=True,
message = (
'sequence not as expected:\n'
'\n'
'same:\n'
'[[1, 2, 3]]\n'
'\n'
'first:\n'
'[[1, 2, 3]]\n'
'\n'
'second:\n'
'[<MyList:[1, 2, 3]>]\n'
'\n'
f"While comparing [1]: <AlreadySeen for [1, 2, 3] at ... "
f"(<class 'testfixtures.comparison.AlreadySeen'>) != "
f"<MyList:[1, 2, 3]> ({type_repr})"
)
)
class TestIgnore(CompareHelper):
class Parent(object):
def __init__(self, id, other):
self.id = id
self.other = other
def __repr__(self):
return '<{}:{}>'.format(type(self).__name__, self.id)
class Child(Parent): pass
def test_ignore_attributes(self):
compare(self.Parent(1, 3), self.Parent(2, 3), ignore_attributes={'id'})
def test_ignore_attributes_different_types(self):
self.check_raises(
self.Parent(1, 3),
self.Child(2, 3),
'<Parent:1> != <Child:2>',
ignore_attributes={'id'}
)
def test_ignore_attributes_per_type(self):
ignore = {self.Parent: {'id'}}
compare(self.Parent(1, 3), self.Parent(2, 3), ignore_attributes=ignore)
self.check_raises(
self.Child(1, 3),
self.Child(2, 3),
'Child not as expected:\n'
'\n'
'attributes same:\n'
"['other']\n"
'\n'
'attributes differ:\n'
"'id': 1 != 2",
ignore_attributes=ignore
)
class TestCompareObject(object):
class Thing(object):
def __init__(self, **kw):
for k, v in kw.items():
setattr(self, k, v)
def test_ignore(self):
def compare_thing(x, y, context):
return compare_object(x, y, context, ignore_attributes=['y'])
compare(self.Thing(x=1, y=2), self.Thing(x=1, y=3),
comparers={self.Thing: compare_thing})
def test_ignore_dict_context_list_param(self):
def compare_thing(x, y, context):
return compare_object(x, y, context, ignore_attributes=['y'])
compare(self.Thing(x=1, y=2, z=3), self.Thing(x=1, y=4, z=5),
comparers={self.Thing: compare_thing},
ignore_attributes={self.Thing: ['z']})
def test_ignore_list_context_list_param(self):
def compare_thing(x, y, context):
return compare_object(x, y, context, ignore_attributes=['y'])
compare(self.Thing(x=1, y=2, z=3), self.Thing(x=1, y=4, z=5),
comparers={self.Thing: compare_thing},
ignore_attributes=['z'])
class BaseClass(ABC):
pass
class MyDerivedClass(BaseClass):
def __init__(self, thing):
self.thing = thing
class ConcreteBaseClass(object): pass
class ConcreteDerivedClass(ConcreteBaseClass):
def __init__(self, thing):
self.thing = thing
class TestBaseClasses(CompareHelper):
def test_abc_equal(self):
thing1 = MyDerivedClass(1)
thing2 = MyDerivedClass(1)
compare(thing1, thing2)
def test_abc_unequal(self):
thing1 = MyDerivedClass(1)
thing2 = MyDerivedClass(2)
self.check_raises(thing1, thing2, message=(
"MyDerivedClass not as expected:\n\n"
"attributes differ:\n"
"'thing': 1 != 2"
))
def test_concrete_equal(self):
thing1 = ConcreteDerivedClass(1)
thing2 = ConcreteDerivedClass(1)
compare(thing1, thing2)
def test_concrete_unequal(self):
thing1 = ConcreteDerivedClass(1)
thing2 = ConcreteDerivedClass(2)
self.check_raises(thing1, thing2, message=(
"ConcreteDerivedClass not as expected:\n\n"
"attributes differ:\n"
"'thing': 1 != 2"
))