mirror of https://github.com/jab/bidict.git
243 lines
5.6 KiB
Plaintext
243 lines
5.6 KiB
Plaintext
# 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/.
|
|
|
|
Test script for bidict.bidict::
|
|
|
|
>>> from bidict import bidict
|
|
>>> keys = (1, 2, 3)
|
|
>>> vals = ('one', 'two', 'three')
|
|
>>> bi = bidict(zip(keys, vals))
|
|
>>> bi == bidict({1: 'one', 2: 'two', 3: 'three'})
|
|
True
|
|
|
|
Works like dict for getting and changing forward mappings::
|
|
|
|
>>> bi[2]
|
|
'two'
|
|
>>> bi[2] = 'twain'
|
|
>>> bi[2]
|
|
'twain'
|
|
>>> bi[4]
|
|
Traceback (most recent call last):
|
|
...
|
|
KeyError: 4
|
|
>>> del bi[2]
|
|
>>> bi.pop(3)
|
|
'three'
|
|
>>> bi
|
|
bidict({1: 'one'})
|
|
|
|
``put`` can also be used to insert a mapping as long as its key and value
|
|
don't already exist::
|
|
|
|
>>> bi.put(0, 'zero')
|
|
>>> bi[0]
|
|
'zero'
|
|
>>> bi.put(1, 'aught')
|
|
Traceback (most recent call last):
|
|
...
|
|
KeyDuplicationError: 1
|
|
>>> del bi[1]
|
|
>>> bi.put(1, 'aught')
|
|
>>> bi[1]
|
|
'aught'
|
|
>>> del bi[0]
|
|
>>> bi
|
|
bidict({1: 'aught'})
|
|
|
|
bidicts maintain references to their inverses via the ``inv`` property,
|
|
which can also be used to access or modify them::
|
|
|
|
>>> bi.inv
|
|
bidict({'aught': 1})
|
|
>>> bi.inv['aught']
|
|
1
|
|
>>> bi.inv['aught'] = 'one'
|
|
>>> bi
|
|
bidict({'one': 'aught'})
|
|
>>> bi.inv.pop('aught')
|
|
'one'
|
|
>>> bi == bi.inv == bidict()
|
|
True
|
|
>>> bi.inv.update(one=1)
|
|
>>> bi
|
|
bidict({1: 'one'})
|
|
>>> bi is bi.inv.inv
|
|
True
|
|
>>> bi.inv is bi.inv.inv.inv
|
|
True
|
|
|
|
bidicts work with ``inverted`` as expected::
|
|
|
|
>>> from bidict import inverted
|
|
>>> biinv = bidict(inverted(bi))
|
|
>>> biinv
|
|
bidict({'one': 1})
|
|
|
|
This created a new object (equivalent but not identical)::
|
|
|
|
>>> biinv == bi.inv
|
|
True
|
|
>>> biinv is bi.inv
|
|
False
|
|
|
|
Inverting the inverse should round-trip::
|
|
|
|
>>> bi == bidict(inverted(inverted(bi)))
|
|
True
|
|
>>> bi = bi.inv
|
|
>>> bi == bidict(inverted(inverted(bi)))
|
|
True
|
|
|
|
The rest of the ``MutableMapping`` interface is supported::
|
|
|
|
>>> bi.get('one')
|
|
1
|
|
>>> bi.get('zero')
|
|
>>> bi.get('zero', 'default')
|
|
'default'
|
|
>>> list(bi.keys())
|
|
['one']
|
|
>>> list(bi.values())
|
|
[1]
|
|
>>> list(bi.items())
|
|
[('one', 1)]
|
|
>>> bi.setdefault('one', 2)
|
|
1
|
|
>>> bi.setdefault('two', 2)
|
|
2
|
|
>>> bi.pop('one')
|
|
1
|
|
>>> bi
|
|
bidict({'two': 2})
|
|
>>> bi.inv
|
|
bidict({2: 'two'})
|
|
>>> bi.pop('wrong', 'number', 'of', 'args')
|
|
Traceback (most recent call last):
|
|
...
|
|
TypeError: pop expected at most 2 arguments (got 4)
|
|
>>> bi.popitem()
|
|
('two', 2)
|
|
>>> bi.popitem()
|
|
Traceback (most recent call last):
|
|
...
|
|
KeyError: 'popitem(): bidict is empty'
|
|
>>> bi.inv.setdefault(3, 'three')
|
|
'three'
|
|
>>> bi
|
|
bidict({'three': 3})
|
|
>>> len(bi) # calls __len__
|
|
1
|
|
>>> [key for key in bi] # calls __iter__, returns keys like dict
|
|
['three']
|
|
>>> 'three' in bi # calls __contains__
|
|
True
|
|
>>> list(bi.keys())
|
|
['three']
|
|
>>> list(bi.values())
|
|
[3]
|
|
>>> bi.update([('four', 4)])
|
|
>>> bi.update({'five': 5}, six=6, seven=7)
|
|
>>> sorted(bi.items(), key=lambda x: x[1])
|
|
[('three', 3), ('four', 4), ('five', 5), ('six', 6), ('seven', 7)]
|
|
>>> bi.clear()
|
|
>>> bi
|
|
bidict()
|
|
|
|
Empty update is a no-op::
|
|
|
|
>>> bi.update()
|
|
>>> bi
|
|
bidict()
|
|
|
|
Not part of the public API, but test this anyway for the coverage::
|
|
|
|
>>> bi._update(False, None)
|
|
>>> bi
|
|
bidict()
|
|
|
|
Initializing with different keys mapping to the same value fails::
|
|
|
|
>>> bidict([(1, 1), (2, 1)])
|
|
Traceback (most recent call last):
|
|
...
|
|
ValueDuplicationError: 1
|
|
|
|
Adding a new key associated with an existing value fails::
|
|
|
|
>>> b = bidict({1: 1})
|
|
>>> b[2] = 1
|
|
Traceback (most recent call last):
|
|
...
|
|
ValueDuplicationError: 1
|
|
>>> b.update({2: 1})
|
|
Traceback (most recent call last):
|
|
...
|
|
ValueDuplicationError: 1
|
|
|
|
``forceput`` and ``forceupdate`` can be used instead::
|
|
|
|
>>> b.forceput(2, 1)
|
|
>>> b
|
|
bidict({2: 1})
|
|
>>> b.forceupdate({1: 1})
|
|
>>> b
|
|
bidict({1: 1})
|
|
|
|
Trying to insert an existing mapping does not raise, and is a no-op::
|
|
|
|
>>> b = bidict({1: 'one'})
|
|
>>> b[1] = 'one'
|
|
>>> b[1]
|
|
'one'
|
|
>>> b.inv['one'] = 1
|
|
>>> b.inv['one']
|
|
1
|
|
|
|
The following case does not half-succeed,
|
|
i.e. the bidict is not in an inconsistent state after::
|
|
|
|
>>> b = bidict(one=1, two=2)
|
|
>>> b['one'] = 2
|
|
Traceback (most recent call last):
|
|
...
|
|
KeyAndValueDuplicationError: ('one', 2)
|
|
>>> len(b) == len(b.inv)
|
|
True
|
|
|
|
``put`` and ``putall`` allow you to have
|
|
per-call control over duplication behavior
|
|
(see doctests in ``../docs/unique-values.rst.inc``).
|
|
|
|
Even with RAISE duplication behavior,
|
|
inserting existing items is a no-op (i.e. it doesn't raise)::
|
|
|
|
>>> from bidict import RAISE
|
|
>>> b.putall([('three', 3), ('one', 1)],
|
|
... on_dup_key=RAISE, on_dup_val=RAISE) is not 'an error'
|
|
True
|
|
>>> b0 = b.copy()
|
|
>>> b.putall([]) # no-op
|
|
>>> b == b0
|
|
True
|
|
|
|
Python 2 dict "view*" APIs are supported::
|
|
|
|
>>> from bidict.compat import PY2
|
|
>>> sorted(b.viewkeys()) == sorted(b.keys()) if PY2 else True
|
|
True
|
|
>>> sorted(b.viewvalues()) == sorted(b.values()) if PY2 else True
|
|
True
|
|
>>> sorted(b.viewitems()) == sorted(b.items()) if PY2 else True
|
|
True
|
|
|
|
Make sure copy.copy ends up calling BidictBase.__copy__
|
|
(should show up in the coverage report)::
|
|
|
|
>>> from copy import copy
|
|
>>> copy(bidict())
|
|
bidict()
|