Passed
Pull Request — dev (#30)
by Konstantinos
03:04 queued 01:34
created

test_subclass_registry()   A

Complexity

Conditions 2

Size

Total Lines 15
Code Lines 9

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 2
eloc 9
nop 1
dl 0
loc 15
rs 9.95
c 0
b 0
f 0
1
import pytest
2
3
4
@pytest.fixture
5
def subclass_registry_metaclass():
6
    from so_magic.utils import SubclassRegistry
7
    return SubclassRegistry
8
9
10
@pytest.fixture
11
def assert_correct_class_creation(subclass_registry_metaclass):
12
    def assert_class_creation(user_class):
13
        assert type(user_class) == subclass_registry_metaclass
14
        assert hasattr(user_class, 'subclasses')
15
        assert hasattr(user_class, 'create')
16
        assert hasattr(user_class, 'register_as_subclass')
17
        assert user_class.subclasses == {}
18
    return assert_class_creation
19
20
21
@pytest.fixture
22
def register_class(subclass_registry_metaclass):
23
    def _register_class(subclass_id: str, inherit=False):
24
        class ParentClass(metaclass=subclass_registry_metaclass):
25
            pass
26
27
        if inherit:
28
            @ParentClass.register_as_subclass(subclass_id)
29
            class Child(ParentClass):
30
                pass
31
        else:
32
            @ParentClass.register_as_subclass(subclass_id)
33
            class Child:
34
                pass
35
36
        child_instance = ParentClass.create(subclass_id)
37
38
        return {'class_registry': ParentClass, 'child': Child, 'child_instance': child_instance}
39
    return _register_class
40
41
42
@pytest.fixture
43
def use_metaclass(register_class, assert_correct_metaclass_behaviour):
44
    is_instance = {True: lambda classes: isinstance(classes['child_instance'], classes['class_registry']),
45
                False: lambda classes: not isinstance(classes['child_instance'], classes['class_registry'])}
46
    def _use_metaclass_in_scenario(subclass_id: str, inherit=False):
47
        classes = register_class(subclass_id, inherit=inherit)
48
        assert_correct_metaclass_behaviour(classes, subclass_id)
49
        assert is_instance[inherit]
50
        return classes['child_instance'], classes['child'], classes['class_registry']
51
    return _use_metaclass_in_scenario
52
53
54
@pytest.fixture
55
def assert_correct_metaclass_behaviour():
56
    def assert_metaclass_behaviour(classes, subclass_id):
57
        assert classes['class_registry'].subclasses[subclass_id] == classes['child']
58
        assert type(classes['child_instance']) == classes['child']
59
        assert isinstance(classes['child_instance'], classes['child'])
60
    return assert_metaclass_behaviour
61
62
63
def test_metaclass_usage(subclass_registry_metaclass, assert_correct_class_creation):
64
    class ParentClass(metaclass=subclass_registry_metaclass):
65
        pass
66
67
    assert_correct_class_creation(ParentClass)
68
69
70
def test_subclass_registry(use_metaclass):
71
    child1_instance1, Child1, ParentClass = use_metaclass('child1', inherit=True)
72
73
    non_existent_identifier = 'child2'
74
75
    exception_message_regex = \
76
        f'Bad "{str(ParentClass.__name__)}" subclass request; requested subclass with identifier ' \
77
        f'{non_existent_identifier}, but known identifiers are ' \
78
        rf'\[{", ".join(subclass_identifier for subclass_identifier in ParentClass.subclasses.keys())}\]'
79
80
    with pytest.raises(ValueError, match=exception_message_regex):
81
        ParentClass.create(non_existent_identifier)
82
83
    child1_instance2, Child2, ParentClass2 = use_metaclass('child2', inherit=False)
84
    assert ParentClass.subclasses['child1'] == Child1
85