242 lines
6.4 KiB
Python
242 lines
6.4 KiB
Python
from __future__ import annotations
|
|
|
|
import pytest
|
|
|
|
from daglib.set import DAGSetView
|
|
|
|
|
|
class TestDAGSetInit:
|
|
"""Test DAGSet initialization."""
|
|
|
|
def test_empty(self) -> None:
|
|
s = DAGSetView()
|
|
assert len(s) == 0
|
|
assert set(s) == set()
|
|
assert list(s) == []
|
|
assert tuple(s) == ()
|
|
|
|
def test_from_set(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
assert len(s) == 3
|
|
assert set(s) == {1, 2, 3}
|
|
|
|
def test_from_list(self) -> None:
|
|
s = DAGSetView([1, 2, 3, 2])
|
|
assert len(s) == 3
|
|
assert set(s) == {1, 2, 3}
|
|
|
|
def test_from_tuple(self) -> None:
|
|
s = DAGSetView((4, 5, 6))
|
|
assert len(s) == 3
|
|
assert set(s) == {4, 5, 6}
|
|
|
|
def test_from_none(self) -> None:
|
|
s = DAGSetView(None)
|
|
assert len(s) == 0
|
|
|
|
def test_from_invalid_type(self) -> None:
|
|
with pytest.raises(TypeError):
|
|
DAGSetView(42) # type: ignore
|
|
|
|
|
|
class TestDAGSetBasicOps:
|
|
"""Test basic set operations."""
|
|
|
|
def test_len(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
assert len(s) == 3
|
|
|
|
def test_iter(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
for i, v in enumerate(s):
|
|
assert v == i + 1
|
|
|
|
def test_contains(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
assert 1 in s
|
|
assert 2 in s
|
|
assert 3 in s
|
|
assert 4 not in s
|
|
|
|
def test_add(self) -> None:
|
|
s = DAGSetView()
|
|
s.add(1)
|
|
assert 1 in s
|
|
assert len(s) == 1
|
|
|
|
def test_discard(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s.discard(2)
|
|
assert 2 not in s
|
|
assert len(s) == 2
|
|
|
|
def test_discard_missing(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s.discard(99)
|
|
assert len(s) == 3
|
|
|
|
def test_remove(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s.remove(2)
|
|
assert 2 not in s
|
|
|
|
def test_remove_missing(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
with pytest.raises(KeyError):
|
|
s.remove(99)
|
|
|
|
|
|
class TestDAGSetCallbacks:
|
|
"""Test callback mechanisms."""
|
|
|
|
class TestBasicOps:
|
|
def test_on_add_callback(self) -> None:
|
|
added: list[int] = []
|
|
s = DAGSetView({1, 2})
|
|
s.on_add = lambda v: added.append(v)
|
|
s.add(3)
|
|
assert 3 in added
|
|
assert len(added) == 1
|
|
|
|
def test_on_remove_callback(self) -> None:
|
|
removed: list[int] = []
|
|
s = DAGSetView({1, 2, 3})
|
|
s.on_remove = lambda v: removed.append(v)
|
|
s.discard(2)
|
|
assert 2 in removed
|
|
assert len(removed) == 1
|
|
|
|
def test_callbacks_none_by_default(self) -> None:
|
|
s = DAGSetView()
|
|
assert s.on_add is None
|
|
assert s.on_remove is None
|
|
|
|
class TestInPlaceOperators:
|
|
def test_ior_triggers_callbacks(self) -> None:
|
|
added: list[int] = []
|
|
s = DAGSetView({1})
|
|
s.on_add = lambda v: added.append(v)
|
|
s |= {2, 3}
|
|
assert 2 in added
|
|
assert 3 in added
|
|
|
|
def test_iadd_triggers_callbacks(self) -> None:
|
|
added: list[int] = []
|
|
s = DAGSetView({1})
|
|
s.on_add = lambda v: added.append(v)
|
|
s += [2, 3]
|
|
assert 2 in added
|
|
assert 3 in added
|
|
|
|
def test_isub_triggers_callbacks(self) -> None:
|
|
removed: list[int] = []
|
|
s = DAGSetView({1, 2, 3})
|
|
s.on_remove = lambda v: removed.append(v)
|
|
s -= {2, 3}
|
|
assert 2 in removed
|
|
assert 3 in removed
|
|
|
|
|
|
class TestDAGSetInPlaceOperators:
|
|
"""Test in-place set operators."""
|
|
|
|
class TestIOR:
|
|
def test_set(self) -> None:
|
|
s = DAGSetView({1, 2})
|
|
s |= {3, 4}
|
|
assert set(s) == {1, 2, 3, 4}
|
|
|
|
def test_list(self) -> None:
|
|
s = DAGSetView({1})
|
|
s |= [2, 3]
|
|
assert set(s) == {1, 2, 3}
|
|
|
|
def test_string(self) -> None:
|
|
s = DAGSetView({"abc", "def"})
|
|
s |= "xyz"
|
|
assert "xyz" in s
|
|
|
|
def test_dagset(self) -> None:
|
|
s = DAGSetView({1, 2})
|
|
other = DAGSetView({3, 4})
|
|
s |= other
|
|
assert set(s) == {1, 2, 3, 4}
|
|
|
|
class TestIAdd:
|
|
def test_set(self) -> None:
|
|
s = DAGSetView({1, 2})
|
|
s += {3, 4}
|
|
assert set(s) == {1, 2, 3, 4}
|
|
|
|
def test_list(self) -> None:
|
|
s = DAGSetView({1})
|
|
s += [2, 3]
|
|
assert set(s) == {1, 2, 3}
|
|
|
|
def test_string(self) -> None:
|
|
s = DAGSetView({"a"})
|
|
s += "b"
|
|
assert "b" in s
|
|
|
|
def test_dagset(self) -> None:
|
|
s = DAGSetView({1, 2})
|
|
other = DAGSetView({3, 4})
|
|
s += other
|
|
assert set(s) == {1, 2, 3, 4}
|
|
|
|
class TestISub:
|
|
def test_set(self) -> None:
|
|
s = DAGSetView({1, 2, 3, 4})
|
|
s -= {2, 3}
|
|
assert set(s) == {1, 4}
|
|
|
|
def test_list(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s -= [2]
|
|
assert set(s) == {1, 3}
|
|
|
|
def test_string(self) -> None:
|
|
s = DAGSetView({"a", "b", "c"})
|
|
s -= "b"
|
|
assert set(s) == {"a", "c"}
|
|
|
|
def test_dagset(self) -> None:
|
|
s = DAGSetView({1, 2, 3, 4})
|
|
other = DAGSetView({2, 3})
|
|
s -= other
|
|
assert set(s) == {1, 4}
|
|
|
|
class TestIAnd:
|
|
def test_set(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s &= {2, 3, 4}
|
|
assert set(s) == {2, 3}
|
|
|
|
def test_list(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s &= [2, 3]
|
|
assert set(s) == {2, 3}
|
|
|
|
def test_dagset(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
other = DAGSetView({2, 3, 4})
|
|
s &= other
|
|
assert set(s) == {2, 3}
|
|
|
|
class TestIXor:
|
|
def test_set(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
s ^= {3, 4, 5}
|
|
assert set(s) == {1, 2, 4, 5}
|
|
|
|
def test_list(self) -> None:
|
|
s = DAGSetView({1, 2})
|
|
s ^= [2, 3]
|
|
assert set(s) == {1, 3}
|
|
|
|
def test_dagset(self) -> None:
|
|
s = DAGSetView({1, 2, 3})
|
|
other = DAGSetView({3, 4, 5})
|
|
s ^= other
|
|
assert set(s) == {1, 2, 4, 5}
|