244 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Python
		
	
	
| import pytest
 | |
| from pytest import approx
 | |
| 
 | |
| from pybind11_tests import ConstructorStats
 | |
| from pybind11_tests import sequences_and_iterators as m
 | |
| 
 | |
| 
 | |
| def test_slice_constructors():
 | |
|     assert m.make_forward_slice_size_t() == slice(0, -1, 1)
 | |
|     assert m.make_reversed_slice_object() == slice(None, None, -1)
 | |
| 
 | |
| 
 | |
| @pytest.mark.skipif(not m.has_optional, reason="no <optional>")
 | |
| def test_slice_constructors_explicit_optional():
 | |
|     assert m.make_reversed_slice_size_t_optional() == slice(None, None, -1)
 | |
|     assert m.make_reversed_slice_size_t_optional_verbose() == slice(None, None, -1)
 | |
| 
 | |
| 
 | |
| def test_generalized_iterators():
 | |
|     assert list(m.IntPairs([(1, 2), (3, 4), (0, 5)]).nonzero()) == [(1, 2), (3, 4)]
 | |
|     assert list(m.IntPairs([(1, 2), (2, 0), (0, 3), (4, 5)]).nonzero()) == [(1, 2)]
 | |
|     assert list(m.IntPairs([(0, 3), (1, 2), (3, 4)]).nonzero()) == []
 | |
| 
 | |
|     assert list(m.IntPairs([(1, 2), (3, 4), (0, 5)]).nonzero_keys()) == [1, 3]
 | |
|     assert list(m.IntPairs([(1, 2), (2, 0), (0, 3), (4, 5)]).nonzero_keys()) == [1]
 | |
|     assert list(m.IntPairs([(0, 3), (1, 2), (3, 4)]).nonzero_keys()) == []
 | |
| 
 | |
|     assert list(m.IntPairs([(1, 2), (3, 4), (0, 5)]).nonzero_values()) == [2, 4]
 | |
|     assert list(m.IntPairs([(1, 2), (2, 0), (0, 3), (4, 5)]).nonzero_values()) == [2]
 | |
|     assert list(m.IntPairs([(0, 3), (1, 2), (3, 4)]).nonzero_values()) == []
 | |
| 
 | |
|     # __next__ must continue to raise StopIteration
 | |
|     it = m.IntPairs([(0, 0)]).nonzero()
 | |
|     for _ in range(3):
 | |
|         with pytest.raises(StopIteration):
 | |
|             next(it)
 | |
| 
 | |
|     it = m.IntPairs([(0, 0)]).nonzero_keys()
 | |
|     for _ in range(3):
 | |
|         with pytest.raises(StopIteration):
 | |
|             next(it)
 | |
| 
 | |
| 
 | |
| def test_nonref_iterators():
 | |
|     pairs = m.IntPairs([(1, 2), (3, 4), (0, 5)])
 | |
|     assert list(pairs.nonref()) == [(1, 2), (3, 4), (0, 5)]
 | |
|     assert list(pairs.nonref_keys()) == [1, 3, 0]
 | |
|     assert list(pairs.nonref_values()) == [2, 4, 5]
 | |
| 
 | |
| 
 | |
| def test_generalized_iterators_simple():
 | |
|     assert list(m.IntPairs([(1, 2), (3, 4), (0, 5)]).simple_iterator()) == [
 | |
|         (1, 2),
 | |
|         (3, 4),
 | |
|         (0, 5),
 | |
|     ]
 | |
|     assert list(m.IntPairs([(1, 2), (3, 4), (0, 5)]).simple_keys()) == [1, 3, 0]
 | |
|     assert list(m.IntPairs([(1, 2), (3, 4), (0, 5)]).simple_values()) == [2, 4, 5]
 | |
| 
 | |
| 
 | |
| def test_iterator_referencing():
 | |
|     """Test that iterators reference rather than copy their referents."""
 | |
|     vec = m.VectorNonCopyableInt()
 | |
|     vec.append(3)
 | |
|     vec.append(5)
 | |
|     assert [int(x) for x in vec] == [3, 5]
 | |
|     # Increment everything to make sure the referents can be mutated
 | |
|     for x in vec:
 | |
|         x.set(int(x) + 1)
 | |
|     assert [int(x) for x in vec] == [4, 6]
 | |
| 
 | |
|     vec = m.VectorNonCopyableIntPair()
 | |
|     vec.append([3, 4])
 | |
|     vec.append([5, 7])
 | |
|     assert [int(x) for x in vec.keys()] == [3, 5]
 | |
|     assert [int(x) for x in vec.values()] == [4, 7]
 | |
|     for x in vec.keys():
 | |
|         x.set(int(x) + 1)
 | |
|     for x in vec.values():
 | |
|         x.set(int(x) + 10)
 | |
|     assert [int(x) for x in vec.keys()] == [4, 6]
 | |
|     assert [int(x) for x in vec.values()] == [14, 17]
 | |
| 
 | |
| 
 | |
| def test_sliceable():
 | |
|     sliceable = m.Sliceable(100)
 | |
|     assert sliceable[::] == (0, 100, 1)
 | |
|     assert sliceable[10::] == (10, 100, 1)
 | |
|     assert sliceable[:10:] == (0, 10, 1)
 | |
|     assert sliceable[::10] == (0, 100, 10)
 | |
|     assert sliceable[-10::] == (90, 100, 1)
 | |
|     assert sliceable[:-10:] == (0, 90, 1)
 | |
|     assert sliceable[::-10] == (99, -1, -10)
 | |
|     assert sliceable[50:60:1] == (50, 60, 1)
 | |
|     assert sliceable[50:60:-1] == (50, 60, -1)
 | |
| 
 | |
| 
 | |
| def test_sequence():
 | |
|     cstats = ConstructorStats.get(m.Sequence)
 | |
| 
 | |
|     s = m.Sequence(5)
 | |
|     assert cstats.values() == ["of size", "5"]
 | |
| 
 | |
|     assert "Sequence" in repr(s)
 | |
|     assert len(s) == 5
 | |
|     assert s[0] == 0 and s[3] == 0
 | |
|     assert 12.34 not in s
 | |
|     s[0], s[3] = 12.34, 56.78
 | |
|     assert 12.34 in s
 | |
|     assert s[0] == approx(12.34, rel=1e-05)
 | |
|     assert s[3] == approx(56.78, rel=1e-05)
 | |
| 
 | |
|     rev = reversed(s)
 | |
|     assert cstats.values() == ["of size", "5"]
 | |
| 
 | |
|     rev2 = s[::-1]
 | |
|     assert cstats.values() == ["of size", "5"]
 | |
| 
 | |
|     it = iter(m.Sequence(0))
 | |
|     for _ in range(3):  # __next__ must continue to raise StopIteration
 | |
|         with pytest.raises(StopIteration):
 | |
|             next(it)
 | |
|     assert cstats.values() == ["of size", "0"]
 | |
| 
 | |
|     expected = [0, 56.78, 0, 0, 12.34]
 | |
|     assert rev == approx(expected, rel=1e-05)
 | |
|     assert rev2 == approx(expected, rel=1e-05)
 | |
|     assert rev == rev2
 | |
| 
 | |
|     rev[0::2] = m.Sequence([2.0, 2.0, 2.0])
 | |
|     assert cstats.values() == ["of size", "3", "from std::vector"]
 | |
| 
 | |
|     assert rev == approx([2, 56.78, 2, 0, 2], rel=1e-05)
 | |
| 
 | |
|     assert cstats.alive() == 4
 | |
|     del it
 | |
|     assert cstats.alive() == 3
 | |
|     del s
 | |
|     assert cstats.alive() == 2
 | |
|     del rev
 | |
|     assert cstats.alive() == 1
 | |
|     del rev2
 | |
|     assert cstats.alive() == 0
 | |
| 
 | |
|     assert cstats.values() == []
 | |
|     assert cstats.default_constructions == 0
 | |
|     assert cstats.copy_constructions == 0
 | |
|     assert cstats.move_constructions >= 1
 | |
|     assert cstats.copy_assignments == 0
 | |
|     assert cstats.move_assignments == 0
 | |
| 
 | |
| 
 | |
| def test_sequence_length():
 | |
|     """#2076: Exception raised by len(arg) should be propagated"""
 | |
| 
 | |
|     class BadLen(RuntimeError):
 | |
|         pass
 | |
| 
 | |
|     class SequenceLike:
 | |
|         def __getitem__(self, i):
 | |
|             return None
 | |
| 
 | |
|         def __len__(self):
 | |
|             raise BadLen()
 | |
| 
 | |
|     with pytest.raises(BadLen):
 | |
|         m.sequence_length(SequenceLike())
 | |
| 
 | |
|     assert m.sequence_length([1, 2, 3]) == 3
 | |
|     assert m.sequence_length("hello") == 5
 | |
| 
 | |
| 
 | |
| def test_map_iterator():
 | |
|     sm = m.StringMap({"hi": "bye", "black": "white"})
 | |
|     assert sm["hi"] == "bye"
 | |
|     assert len(sm) == 2
 | |
|     assert sm["black"] == "white"
 | |
| 
 | |
|     with pytest.raises(KeyError):
 | |
|         assert sm["orange"]
 | |
|     sm["orange"] = "banana"
 | |
|     assert sm["orange"] == "banana"
 | |
| 
 | |
|     expected = {"hi": "bye", "black": "white", "orange": "banana"}
 | |
|     for k in sm:
 | |
|         assert sm[k] == expected[k]
 | |
|     for k, v in sm.items():
 | |
|         assert v == expected[k]
 | |
|     assert list(sm.values()) == [expected[k] for k in sm]
 | |
| 
 | |
|     it = iter(m.StringMap({}))
 | |
|     for _ in range(3):  # __next__ must continue to raise StopIteration
 | |
|         with pytest.raises(StopIteration):
 | |
|             next(it)
 | |
| 
 | |
| 
 | |
| def test_python_iterator_in_cpp():
 | |
|     t = (1, 2, 3)
 | |
|     assert m.object_to_list(t) == [1, 2, 3]
 | |
|     assert m.object_to_list(iter(t)) == [1, 2, 3]
 | |
|     assert m.iterator_to_list(iter(t)) == [1, 2, 3]
 | |
| 
 | |
|     with pytest.raises(TypeError) as excinfo:
 | |
|         m.object_to_list(1)
 | |
|     assert "object is not iterable" in str(excinfo.value)
 | |
| 
 | |
|     with pytest.raises(TypeError) as excinfo:
 | |
|         m.iterator_to_list(1)
 | |
|     assert "incompatible function arguments" in str(excinfo.value)
 | |
| 
 | |
|     def bad_next_call():
 | |
|         raise RuntimeError("py::iterator::advance() should propagate errors")
 | |
| 
 | |
|     with pytest.raises(RuntimeError) as excinfo:
 | |
|         m.iterator_to_list(iter(bad_next_call, None))
 | |
|     assert str(excinfo.value) == "py::iterator::advance() should propagate errors"
 | |
| 
 | |
|     lst = [1, None, 0, None]
 | |
|     assert m.count_none(lst) == 2
 | |
|     assert m.find_none(lst) is True
 | |
|     assert m.count_nonzeros({"a": 0, "b": 1, "c": 2}) == 2
 | |
| 
 | |
|     r = range(5)
 | |
|     assert all(m.tuple_iterator(tuple(r)))
 | |
|     assert all(m.list_iterator(list(r)))
 | |
|     assert all(m.sequence_iterator(r))
 | |
| 
 | |
| 
 | |
| def test_iterator_passthrough():
 | |
|     """#181: iterator passthrough did not compile"""
 | |
|     from pybind11_tests.sequences_and_iterators import iterator_passthrough
 | |
| 
 | |
|     values = [3, 5, 7, 9, 11, 13, 15]
 | |
|     assert list(iterator_passthrough(iter(values))) == values
 | |
| 
 | |
| 
 | |
| def test_iterator_rvp():
 | |
|     """#388: Can't make iterators via make_iterator() with different r/v policies"""
 | |
|     import pybind11_tests.sequences_and_iterators as m
 | |
| 
 | |
|     assert list(m.make_iterator_1()) == [1, 2, 3]
 | |
|     assert list(m.make_iterator_2()) == [1, 2, 3]
 | |
|     assert not isinstance(m.make_iterator_1(), type(m.make_iterator_2()))
 |