Python unittest - asserting dictionary with lists

2019-02-12 19:13发布

While writing some tests for my class, I encountered interesting simple problem. I would like to assertDictEqual two dictionaries containing some list. But this lists may not be sorted in a same way -> which results in failed test

Example:

def test_myobject_export_into_dictionary(self):
    obj = MyObject()
    resulting_dictionary = {
            'state': 2347,
            'neighbours': [1,2,3]
        }
    self.assertDictEqual(resulting_dictionary, obj.exportToDict())

This fail from time to time, depending on order of elements in list

FAIL: test_myobject_export_into_dictionary
------------------------------------
-  'neighbours': [1,2,3],
+  'neighbours': [1,3,2],

Any ideas how to assert this in a simple way?

I was thinking about using set instead of list or sorting lists before comparison.

4条回答
唯我独甜
2楼-- · 2019-02-12 19:20

How about using all:

assert all( (k,v) in resulting_dictionary.iteritems() 
            for (k,v) in obj.exportToDict().iteritems() )

I use something like this with py.test, but I think it should work for you.


A commenter pointed out that the order will screw me here---fair enough...I'd just use sets, then.

查看更多
Bombasti
3楼-- · 2019-02-12 19:31

You might try PyHamcrest (Example corrected)

assert_that(obj.exportToDict(), has_entries(
                                    { 'state': 2347,
                                      'neighbours': contains_inanyorder(1,2,3) }))

(The first value 2347 actually gets wrapped in an implicit equal_to matcher.)

查看更多
兄弟一词,经得起流年.
4楼-- · 2019-02-12 19:38

maybe you can check for the two elements separately:

    obj_dict = obj.exportToDict()
    self.assertEqual(resulting_dictionary['state'], obj_dict['state'])
    self.assertCountEqual(resulting_dictionary['neighbours'], obj_dict['neighbours'])
查看更多
时光不老,我们不散
5楼-- · 2019-02-12 19:42

You can do:

a = {i:sorted(j) if isinstance(j, list) else j for i,j in resulting_dictionary.iteritems()}
b = {i:sorted(j) if isinstance(j, list) else j for i,j in obj.exportToDict().iteritems()}
self.assertDictEqual(a, b)
查看更多
登录 后发表回答